计算一个无符整数中1Bit的个数(1) 2010-04-20 10:52:48

分类: C/C++

[转]计算一个无符整数中1Bit的个数(1)

Count the number of bits that are on in an unsigned integer(计算一个无符整数中1Bit的个数)-- (1)

计算一个无符号整数中有多少的Bit为1

这是一个经常遇到的经典问题,这里分两个部分讲解和总结,首先对讲解现有的算法,然后再讲解一些改进算法。

1.循环法(Iterated Count)

int bitcount (unsigned int n)
{
int count=0;      
    while (n)  {
        count += n & 0x1u ;
        n >>= 1 ;
      }
return count ;
}

最容易理解和想到的方法。对每一位依次判断是否为1,如果是就在count上加1。

循环的次数是常数(n的位数)。在1比较稀疏的时候效率低,可用方法2改进。

2.Bit1稀疏Sparse Ones

int bitcount (unsigned int n)
{
int count=0 ;
         while (n)  {
         count++ ;
         n &= (n - 1) ;
     }
     return count ;
}

理解这个算法的核心,只需理解2个操作:

1> 当一个数被减1时,他最右边的那个值为1的Bit将变为0,同时其右边的所有的Bit都会变成1。
2>“&=”,位与并赋值操作。去掉已经被计数过的1,并将改值重新设置给n.

这个算法循环的次数是bit位为一的个数。也就说有几个Bit为1,循环几次。对Bit为1比较稀疏的数来说,性能很好。如:0x1000 0000, 循环一次就可以。

3.密集1的算法 Dense Ones

   int bitcount (unsigned int n)    

   {

      int count = 8 * sizeof(int) ;

      n ^= (unsigned int) -1 ;

      while (n)

      {

         count-- ;

         n &= (n - 1) ;

      }

      return count ;

   }

与2稀疏1的算法相类似。不同点是,针对1密集的情况,循环的次数会大大减少。他的循环次数:sizeof(int)-Bit 1的个数。

4.8bit静态表查找法 Precompute_8bit

     static int bits_in_char [256] = {             

0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4, 1, 2, 2, 3, 2, 3, 3, 4, 2,

3, 3, 4, 3, 4, 4, 5, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3,

3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3,

4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 2, 3, 3, 4,

3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5,

6, 6, 7, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4,

4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5,

6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 2, 3, 3, 4, 3, 4, 4, 5,

3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 3,

4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 4, 5, 5, 6, 5, 6, 6, 7, 5, 6,

6, 7, 6, 7, 7, 8

};

 
 
     int bitcount (unsigned int n)
     {
        // works only for 32-bit ints
        return bits_in_char [n           & 0xffu]
            +bits_in_char [(n >>8) & 0xffu] //移位>>优先级大于& ,所以()可以去掉。
            +bits_in_char [(n >> 16) & 0xffu]
            +bits_in_char [(n >> 24) & 0xffu] ;
     }

使用静态数组表,列出所有8bit(256个)无符号数含有Bit1的个数。将32Bit 的n分4部分,直接在表中找到对应的Bit1的个数,然后求和。

这是最快的方法了。缺点是需要比较大的内存。

5.16bit静态表查找法Precompute_16bit

因为在计算64位int时,以上方法4并不总是最快,所以有以下的一个进化版,就是用十六Bit的表来作驱动映射。这样需要的内存就更大了。

static char bits_in_16bits [0x1u << 16] …;

int bitcount (unsigned int n)
{
       // works only for 32-bit ints
       return bits_in_16bits [n           & 0xffffu]
           +bits_in_16bits [(n >> 16) & 0xffffu] ;
}

6. 合并计数器法 Parallel Counter

unsigned numbits(unsigned int i)

{
 
      unsigned int const MASK1= 0x55555555;
      unsigned int const MASK2= 0x33333333;
      unsigned int const MASK4= 0x0f0f0f0f;
      unsigned int const MASK8= 0x00ff00ff;
unsigned int const MASK16 = 0x0000ffff;
/*
MASK1= 01010101010101010101010101010101
MASK2= 00110011001100110011001100110011
MASK4= 00001111000011110000111100001111
MASK8= 00000000111111110000000011111111
MASK16 = 00000000000000001111111111111111
*/
 
      
    i = (i&MASK1 ) + (i>>1 &MASK1 );
      i = (i&MASK2 ) + (i>>2 &MASK2 );
      i = (i&MASK4 ) + (i>>4 &MASK4 );
      i = (i&MASK8 ) + (i>>8 &MASK8 );
      i = (i&MASK16) + (i>>16&MASK16);
      
    return i;
}

这个算法是一种合并计数器的策略。把输入数的32Bit当作32个计数器,代表每一位的1个数。然后合并相邻的2个“计数器”,使i成为16个计数器,每个计数器的值就是这2个Bit的1的个数;继续合并相邻的2个“计数器“,使i成为8个计数器,每个计数器的值就是4个Bit的1的个数。。依次类推,直到将i变成一个计数器,那么它的值就是32Bit的i中值为1的Bit的个数。

举个例子,假设输入的i值为10010111011111010101101110101111(十进制2541575087)

计算过程如下:(共22个1)

1.        将32个计数器合并为16个,每一个计数器代表 2-bit 的1个数

1 0 0 1 0 1 1 0 0 0 1 1 1 1 1 1 =1001011000111111

+0 1 1 1 1 1 1 1 1 1 0 1 0 0 1 1 =0111111111010011

----------------------------------------------------------------------

1 1 1 2 1 2 2 1 1 1 1 2 1 1 2 2 = 01 01 01 10 01 10 10 01 01 01 01 10 01 01 10 10

2.        将16个计数器合并为8个,每一个计数器代表 4-bit 的1个数

1 1 1 2 1 1 1 2 =   01   01   01   10   01   01   01   10

+1 2 2 1 1 2 1 2 =   01   10   10   01   01   10   01   10

---------------   ---------------------------------------

2 3 3 3 2 3 2 4 = 0010 0011 0011 0011 0010 0011 0010 0100

3.        将8个计数器合并为4个,每一个计数器代表 8-bit 的1个数

3 3 3 4 =     0010     0011     0010     0010

+2 3 2 2 =     0011     0011     0011     0100

-------   -----------------------------------

5 6 5 6 = 00000101 00000110 00000101 00000110

4.        将4个计数器合并为2个,每一个计数器代表 16-bit 的1个数

55 =         00000101         00000101

+ 66 =         00000110         00000110

-----   ---------------------------------

11 11 = 0000000000001011 0000000000001011

5.        最后,将2个计数器合并为1个,每一个计数器代表 32-bit (也就是输入的值i)的1个数

11 =                 0000000000001011

+11 =                 0000000000001011

--   --------------------------------

22 = 00000000000000000000000000010110

对于该算法的实现,另外有一种比较好的写法,这种算法避免了使用常数宏,使比较通用的实现:

#define TWO(c)       (0x1u << (c))
#define MASK(c)      (((unsigned int)(-1)) / (TWO(TWO(c)) + 1u))
#define COUNT(x,c) ((x) & MASK(c)) + (((x) >> (TWO(c))) & MASK(c))
  int bitcount (unsigned int n)
{
       n = COUNT(n, 0) ;
       n = COUNT(n, 1) ;
       n = COUNT(n, 2) ;
       n = COUNT(n, 3) ;
       n = COUNT(n, 4) ;
       /* n = COUNT(n, 5) ;      for 64-bit integers */
       return n ;
}
 

转载于:https://www.cnblogs.com/jeanschen/p/3550561.html

ZT 计算一个无符整数中1Bit的个数(1) 2010-04-20 10:52:48相关推荐

  1. C语言与,或,非,同或,异或,左移,右移及优先级,查表法计算一个十进制数中1的个数

    1.与:& 2.或:| 3.非:~ 4.异或:^ 5.同或:~^ 6.左移:(<<) 7.右移:(>>) 注意:位运算符中求反运算"~"优先级最高, ...

  2. 本题要求实现一个统计整数中指定数字的个数的简单函数。

    本题要求实现一个统计整数中指定数字的个数的简单函数. 函数接口定义: int CountDigit( int number, int digit ); 其中number是不超过长整型的整数,digit ...

  3. 【FPGA入门十二】1bit全加器实现计算8位二进制数中1的个数

    文章目录 一.实验任务 二.设计思路 三.代码实现 ①设计按键消抖模块 ②设计按键输入8bit二进制数 ③设计计算8bit二进制数中1的个数模块 ④设计数码管显示模块 ⑤顶层模块 ⑥设计仿真文件 ⑦仿 ...

  4. 计算一个字符串里面特定字符的个数

    题目 计算一个字符串中特定字符的个数 思路1 利用for循环,对比,计算,这是很正常的想法.如果不用for循环呢? String test="ababdadwdewddedwwedfewfe ...

  5. 计算数值二进制表达式中1的个数

    --------------------------------------------- -- 时间:2019-03-21 -- 创建人:Ruo_Xiao -- 邮箱:xclsoftware@163 ...

  6. c语言用字符串统计一个整数中数字的个数_全国计算机等级考试二级C语言

    590. 在 C 程序中,用_____表示逻辑值"真". A) 1 B) 非 0 的数 C) 非 1 的数 D) 大于 0 的数 正确答案:B 591. 假设变量已正确定义,表达式 ...

  7. 计算一个姓名文件中出现的所有姓的个数

    有一个name.txt,里面存储了很多人的姓名,且每个人的姓与名都以空格来隔开,试计算出所有的姓的个数: 假设的一个名单: 张 三 李 四 王 五 李 宁 胡 歌 张 学友 郭 富城 刘 德华 黎 明 ...

  8. stream 计算一个List对象中某个字段总和

    int total = list.stream().mapToInt(User::getAge).sum();

  9. 100w个整数中,每个数各不相同,且都小于100w,问如何快速的排序

    之前在网上看过这道题目,及其解法.现记录下来,奉献给大家喽~ 这种方法以牺牲空间为代价,换取O(n)的算法复杂度. (1)取一块长度为100w位的内存,记为A (2)遍历所有的整数i(题目中的100w ...

最新文章

  1. pytorch JIT浅解析
  2. 安卓关于fragment切换后继续运行的问题!
  3. pdf屏幕取词 android,===C#屏幕取词Demo热键版鼠标划词版【含语音版】【附源码】===...
  4. 前后端分离模式下的权限设计方案
  5. 第四范式亮相中关村论坛 共话科技创新与技术发展
  6. 左神算法:在二叉树中找到累加和为指定值的最长路径长度(Java版)
  7. Python二级笔记(10)
  8. IOS UIWindow 和 UIScreen
  9. Python __all__的作用
  10. 死磕算法!精选35篇算法设计实例+6本必读书打包送你
  11. 分寸:切记别把话说得太死 — 《别输在不会表达上》
  12. 服务器压力测试知识集锦
  13. 基于ESP8266芯片的实时温湿度传感器
  14. 基于移动终端的增强现实技术方案(已实现)
  15. kjb文件 解析_在Linux上使用lnav监控和分析Apache日志文件工具
  16. 这段代码不讲武德,劝你耗子尾汁
  17. 实现模糊查询时对特殊字符进行处理和对查询结果进行处理
  18. 程序思路分享 计算机毕业设计Python+Spark+Hadoop+Flink微博舆情预警系统 微博舆情可视化 舆情大数据 微博大数据 微博爬虫 大数据毕业设计 大数据毕设
  19. 2017年全国大学生数学建模B题数据处理
  20. Python版本的温湿度+Nokia5110 display(SPI)

热门文章

  1. Uva10616+10820
  2. Mini-USB、Micro-USB(Micro-B)、Type-C等接口
  3. 关键点检测评价指标OKS
  4. vncviewer退出全屏
  5. mysql 5.6 登录 警告_mysql登录警告问题的解决方法
  6. 基于pyqt5 构建弹窗进度条,在大型计算中实时显示进度
  7. 把视频中人的台词去掉且保留背景音乐的方法
  8. java white case语句_JAVA基础(一)
  9. 因为洋红色被告垄断,德国电信心塞
  10. 26.编辑距离(一)