我有一个长度为8的二进制数,例如00110101设置了8位。 我需要一个快速位计数来确定设置位数。 像这样x = x&(x-1)运行算法会将它限制为包含数字中的设置位数但我不太确定如何使用它。一点点帮助就会明显!!
答案 0 :(得分:4)
此x=x&(x-1)
从二进制字符串中删除最低设置位。如果计算在数字变为0之前删除最低位的次数,则将获得已设置的位数。
char numBits(char x){
char i = 0;
if(x == 0)
return 0;
for(i = 1; x &= x-1; i++);
return i;
}
答案 1 :(得分:3)
执行x = x & (x-1)
将删除最低位集。因此,在您的情况下,迭代将执行,
loop #1: 00110101(53) & 00110100(52) = 00110100(52) :: num bits = 1
loop #2: 00110100(52) & 00110011(51) = 00110000(48) :: num bits = 2
loop #3: 00110000(48) & 00101111(47) = 00100000(32) :: num bits = 3
loop #3: 00100000(32) & 00011111(31) = 00000000( 0) :: num bits = 4
允许的迭代次数将是给定数字中的总位数。
答案 2 :(得分:1)
US Patent 6,516,330 – Counting set bits in data words
(我只是发明了它 - 不要让我解释它。)
答案 3 :(得分:1)
问题中描述的方法(通常归因于K& R)具有n的复杂度,其中n是数字中设置的比特数。
通过使用额外的内存,我们可以将其带到O(1):
使用位数(编译时操作)初始化查找表,然后引用它; 您可以在此处找到此方法:Counting bits set by lookup table
你可以在Henry S. Warren,Jr。(2007)“追求加速人口数”,美丽代码 pp.147-158 O'Reilly中找到不同方法的详细讨论
答案 4 :(得分:1)
我有以下代码片段仅适用于无符号整数: 希望这会有所帮助。
uint G = 8501; //10 0001 0011 0101
uint g = 0;
for (int i = 0; i < 32; i++)
{
g += (G << (31 - (i % 32))) >> 31;
}
Console.WriteLine(g); //6
答案 5 :(得分:0)
取决于你如何计算设置位 - 在我们的例子中,真正设置的位数是4;位宽(我在此定义为最高位组的编号加1)为6(因为您需要6位来表示您的数字)。后者可以用
确定char highestbit (uint32_t num)
{
char i;
for (i = 0; i < sizeof(num)*8; i++) {
if ((1L << i) > num) {
return i;
}
}
return sizeof(num)*8;
}
(未测试的)
答案 6 :(得分:0)
x := (x and $55555555) + ((x shr 1) and $55555555);
x := (x and $33333333) + ((x shr 2) and $33333333);
x := (x and $0F0F0F0F) + ((x shr 4) and $0F0F0F0F);
x := (x and $00FF00FF) + ((x shr 8) and $00FF00FF);
x := (x and $0000FFFF) + ((x shr 16) and $0000FFFF);
取自matrix67的博客,中文版。