算法的强大——快速计算一个正二进制整数中包含多少个1
2015-05-21 15:33
323 查看
原题:一个正整数,转成二进制后,这个二进制数包含多少个1?
这个问题在网上看过多次,几番思考,也没有什么好的办法。采用最基本的办法,逐位判断,是1的统计加1,最后将统计数返回。
以下是这个思路的VB2008代码,不失一般性,将正整数的范围控制在(1~231-1)
Private Function GetCount1OfValue(ByVal Value As
Integer) As Integer
Dim i As
Integer, Count As Integer =
0
For i
= 0 To 30
If (Value And 2
^ i) = 2 ^ i Then Count += 1
Next
Return Count
End Function
但是近日,在网上发现一个很巧妙的算法,能够快速实现上述的计算功能。代码贴于下方
Private Function GetCount1OfValue(ByVal Value As
Integer) As Integer
Dim Count As
Integer = 0
Do While Value
> 0
Value = Value And (Value
- 1)
Count +=1
Loop
Return Count
End Function
这段代码的精髓就是在这一句:Value = Value And (Value - 1)
曾经用过类似语句的在我的博客“判断是否是2的N次方——证明x & (x -
1)==0的正确性”
那么这句语句到底起到什么作用呢?看下面的分析
假设Value=X1X2……Xn-1Xn,其中Xi(1≤i≤n)为1或0
不妨设Xi是最右边的1,那么Value就可以写成如下的形式
Value=X1X2……Xi-1Xi0……0,其中(1≤i≤n),Xi后面有n-i个0
因为Xi=1,所以Value=X1X2……Xi-110……0,其中(1≤i≤n),1后面有n-i个0
则Value-1=X1X2……Xi-101……1,其中(1≤i≤n),0后面有n-i个1
则Value And (Value-1)=X1X2……Xi-100……0,其中(1≤i≤n),Xi-1后面有n-i+1个0
因此,Value And (Value-1)的效果把最右边的1变成0
在上面的代码中,每把最右边的1变成0,则统计数加1,直到所有的1变成0为止。
这两个算法,第一个算法的循环次数是固定的,是31次,无论数值是多少(必须在范围之内)。而第二个算法和Value中的1的个数有关,循环的次数就是1的个数,可见该算法之妙。
作者:万仓一黍
出处:http://grenet.cnblogs.com/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
在大牛里学习了很多东西。一点一点的记录积累知识。
这个问题在网上看过多次,几番思考,也没有什么好的办法。采用最基本的办法,逐位判断,是1的统计加1,最后将统计数返回。
以下是这个思路的VB2008代码,不失一般性,将正整数的范围控制在(1~231-1)
Private Function GetCount1OfValue(ByVal Value As
Integer) As Integer
Dim i As
Integer, Count As Integer =
0
For i
= 0 To 30
If (Value And 2
^ i) = 2 ^ i Then Count += 1
Next
Return Count
End Function
但是近日,在网上发现一个很巧妙的算法,能够快速实现上述的计算功能。代码贴于下方
Private Function GetCount1OfValue(ByVal Value As
Integer) As Integer
Dim Count As
Integer = 0
Do While Value
> 0
Value = Value And (Value
- 1)
Count +=1
Loop
Return Count
End Function
这段代码的精髓就是在这一句:Value = Value And (Value - 1)
曾经用过类似语句的在我的博客“判断是否是2的N次方——证明x & (x -
1)==0的正确性”
那么这句语句到底起到什么作用呢?看下面的分析
假设Value=X1X2……Xn-1Xn,其中Xi(1≤i≤n)为1或0
不妨设Xi是最右边的1,那么Value就可以写成如下的形式
Value=X1X2……Xi-1Xi0……0,其中(1≤i≤n),Xi后面有n-i个0
因为Xi=1,所以Value=X1X2……Xi-110……0,其中(1≤i≤n),1后面有n-i个0
则Value-1=X1X2……Xi-101……1,其中(1≤i≤n),0后面有n-i个1
则Value And (Value-1)=X1X2……Xi-100……0,其中(1≤i≤n),Xi-1后面有n-i+1个0
因此,Value And (Value-1)的效果把最右边的1变成0
在上面的代码中,每把最右边的1变成0,则统计数加1,直到所有的1变成0为止。
这两个算法,第一个算法的循环次数是固定的,是31次,无论数值是多少(必须在范围之内)。而第二个算法和Value中的1的个数有关,循环的次数就是1的个数,可见该算法之妙。
/* 快速计算一个十进制数转换而二进制数时1的个数。 以13348238为例。 二进制为:110010111010110110001110 算法运行:循环14次,14次减法,14次位运算,14次加法运算 算法核心:n = (n&(n - 1)) 【-----------------------------------------】 那么这句语句到底起到什么作用呢?看下面的分析 假设Value=X1X2……Xn-1Xn,其中Xi(1≤i≤n)为1或0 不妨设Xi是最右边的1,那么Value就可以写成如下的形式 Value=X1X2……Xi-1Xi0……0,其中(1≤i≤n),Xi后面有n-i个0 因为Xi=1,所以Value=X1X2……Xi-110……0,其中(1≤i≤n),1后面有n-i个0 则Value-1=X1X2……Xi-101……1,其中(1≤i≤n),0后面有n-i个1 则Value And (Value-1)=X1X2……Xi-100……0,其中(1≤i≤n),Xi-1后面有n-i+1个0 因此,Value And (Value-1)的效果把最右边的1变成0 在上面的代码中,每把最右边的1变成0,则统计数加1,直到所有的1变成0为止。 【---------------------------------------------------------------------------】 优于普通的遍历。 */ #include<iostream> using namespace std; int main() { int Value; while (cin >> Value) { int m = 0; while (Value>0) { Value = (Value&(Value - 1)); m++; } cout << m << endl; } return 0; }
作者:万仓一黍
出处:http://grenet.cnblogs.com/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
在大牛里学习了很多东西。一点一点的记录积累知识。
相关文章推荐
- 算法的强大——快速计算一个正二进制整数中包含多少个1
- 计算一个整数的二进制表示有多少个1(别人的最快算法)
- Java算法总结:输入一个整数,求该整数的二进制表示中有多少个1
- 计算一个整数的二进制中有多少个1
- 计算一个整数二进制表示时有多少位是为1的算法
- 计算将一个十进制整数转换成二进制含多少个1
- 每天学习一算法系列(26)(输入一个整数,求该整数的二进制表达中有多少个1)
- 请设计一个算法,计算n的阶乘有多少个尾随零。 给定一个int n,请返回n的阶乘的尾零个数。保证n为正整数
- 计算将一个十进制整数转换成二进制含多少个1
- 191. Number of 1 Bits (计算一个整数的二进制表示中有多少个1)
- C++ 算法之 输入两个整数m n,求计算需要改变m的二进制表示中的多少位才能得到n
- 数组A包含0到n的所有整数,但其中缺了一个,且数组A里的整数都是用二进制形式存储,求这个缺失的整数
- C#第2次试验(1).创建一个抽象类A,该类中包含一个求两个数之和抽象方法。创建一个子类B,在B中重写求和方法,且使用方法重载使得方法可以分别计算整数、双精度、字符串。
- 10--输入一个十进制的整数,转化为二进制,输出有多少个1
- 打印出一个整数二进制包含1的个数
- 快速计算整数的二进制表示法中1的个数
- (转贴)反转一个字节 和 判断32位整数二进制中1的个数 的算法
- 输入两个整数m和n,计算需要改变m的二进制表示中的多少位才能得到n。
- 1.写一个函数返回参数二进制中1的个数2.获取一个数二进制序列中所有的偶数为和奇数位,分别输出二进序列3.输出一个整数的每一位4.编程实现:两个int(32位)整数m和n的二进制表达中,有多少个位(b
- (转贴)反转一个字节 和 判断32位整数二进制中1的个数 的算法