在 char 数组中查找最多 6 个连续 0 位的最快方法

Fastest way to find up to 6 consecutive 0 bits in a char array

本文关键字:连续 方法 数组 char 查找      更新时间:2023-10-16

这是我目前正在做的事情:

int dataLen = 500;
char data[dataLen];
int desired = 1; // between 1 and 6, inclusive
...
char bits[dataLen * 8];
for (int32 j = 0; j < dataLen; j++) {
for (int i = 0; i < 8; i++) {
bits[j*8+i] = ( (data[j] & (1 << (7-i))) ? '1' : '0' );
}
}
int offset = std::search_n(bits, bits + dataLen*8, desired, '0') - bits;

真的很讨厌,我知道,这是扼杀性能。

查找 char 数组中第一组连续 0 位的第一组x位的位偏移量的最快方法是什么,其中0 < x < 7?我在 GCC 上使用 SSE 4.2,所以像 __builtin_ctz、__builtin_popcountl 这样的内置是一种选择,我只是想不出使用它们的最佳方式。

有多少个数字有 6 个连续的 0 位(即使考虑 2 个连续的字节)?

Byte 1
XXXXXXXX
000000??             0/1/2/3
?000000?             0/1/128/129
??000000             0/64/128/192

因此,如果我们一次考虑 1 个字节,则 0/1/2/3/64/128/192

Byte a   Byte b
XXXXXXXX XXXXXXXX
??100000 0???????    (a & 31 == 0) && (b & 128 == 0)
???10000 00??????    (a & 15 == 0) && (b & 192 == 0)
????1000 000?????    (a & 7  == 0) && (b & 224 == 0)
?????100 0000????    (a & 3  == 0) && (b & 240 == 0)
??????10 00000???    (a & 1  == 0) && (b & 248 == 0)

因此,绝对最多 12 个测试为您提供所有组合。
我敢肯定,如果你聪明地进行比较,你可以减少这种情况。

如果我们在下面@Michael毛刺解决方案中使用表格驱动的方法。然后我们可以组织它,以便您可以为每个字节进行一到两次比较。

struct TestStruct
{
bool    is6Consecutive;
bool    hasTrailing;
int     maskNextByte;
int     offset;
};
TestStruct   testData[256];
std::size_t findOffsetOf_6ConsecutiveZero(char const* data, size_t size)
{
for(int loop = 0;loop < (size-1); ++loop)
{
char const&           val  = data[loop];
TestStructure const&  test = testData[val];
if (test.is6Consecutive)
{   return loop*8 + test.offset;
}
if (test.hasTrailing)
{
if ((data[loop + 1] & test.maskNextByte) == 0)
{   return loop*8 + test.offset;
}
}
}
// Test last byte
TestStructure const&  test = testData[data[size-1]];
if (test.is6Consecutive)
{   return (size-1)*8 + test.offset;
}
return -1;
}

前几个表条目:

TestStruct   testData[256] =
{
{true,  false, 0x00, 0},
{true,  false, 0x00, 0},
{true,  false, 0x00, 0},
{true,  false, 0x00, 0},
{false, true,  0xf0, 6},  // 4 => 00000100 <Next 4 bytes are zero we hit>
{false, false, 0x00, 0},  // 5 => 00000101 <Ignore and move on>
{false, true,  0xf8, 7},  // 6 => 00000110 <Next 5 bytes are zero we hit>
etc...
};

这是一个与问题中提供的函数的输出相匹配的函数(至少在有限的测试下)。 它使用表查找,表是由一次性脚本生成的。 老实说,我不确定它的性能是否与使用位测试黑客或 GCC 内置的建议竞争,但我敢打赌它不会太远。

struct zeros {
unsigned char leading;
unsigned char internal;
unsigned char trailing;
};
// forward declaration so the long, non-interesting table is at the 
//  end of this 
static struct zeros const zero_table[256];
int find_zero_bits_offset( char const* data, int datalen, int desired)
{
int offset = -1;
int found = 0;
char const* dataptr = &data[0];
char const* endptr  = &data[datalen];

// first, find which byte the sequence of zeros begins
while (!found && (dataptr != endptr)) {
unsigned char ch1 = *dataptr++;
unsigned char ch2 = (dataptr != endptr) ? *dataptr : 0xff;
int internal = zero_table[ch1].internal;
int trailing = zero_table[ch1].trailing;
int leading =  zero_table[ch2].leading;
found = (desired <= internal) || 
(trailing && (desired <= (trailing + leading)));
}

// now zero in on where the sequence starts within the byte
if (found) {
char ch = 0;
unsigned int mask = 0;
--dataptr;
// dataptr points to the byte where the sequence of zeros starts.
//  figure out exactly where the sequence is...
// there's possibly some opportunity for optimization, if neccesary,
//  by testing if the sequence was found in the "leading", "internal", or
//  "trailing" cases. But the explicit loop will only iterate at most
//  8 times (and will early-out on the first iteration if the match is 
//  for the "leading" case) that it didn't seem too important
ch = *dataptr;
offset = (dataptr - data) * 8;
// figure out where the appropriate internal run starts
// note that the offset we need to return isn't necessarily the
//  offset for the run of zeros counted by zero_table[tmp].internal_offset
//  since a sufficient shorter run might come first
// there may be a more efficient bithack for this, but the
//  loop will iterate at most 8 times...
mask = ((1 << desired) - 1);
mask <<= (8 - desired);
while ((ch & mask) != 0) {
++offset;
mask >>= 1;
}
}
else {
// however you want to handle the "not found" situation. 
//  This is equivalent to what was in the question:
offset = (endptr - data) * 8;
}
return offset;
}

static struct zeros const zero_table[256] = {
{ 8, 8, 8 },  // 0000 0000
{ 7, 7, 0 },  // 0000 0001
{ 6, 6, 1 },  // 0000 0010
{ 6, 6, 0 },  // 0000 0011
{ 5, 5, 2 },  // 0000 0100
{ 5, 5, 0 },  // 0000 0101
{ 5, 5, 1 },  // 0000 0110
{ 5, 5, 0 },  // 0000 0111
{ 4, 4, 3 },  // 0000 1000
{ 4, 4, 0 },  // 0000 1001
{ 4, 4, 1 },  // 0000 1010
{ 4, 4, 0 },  // 0000 1011
{ 4, 4, 2 },  // 0000 1100
{ 4, 4, 0 },  // 0000 1101
{ 4, 4, 1 },  // 0000 1110
{ 4, 4, 0 },  // 0000 1111
{ 3, 4, 4 },  // 0001 0000
{ 3, 3, 0 },  // 0001 0001
{ 3, 3, 1 },  // 0001 0010
{ 3, 3, 0 },  // 0001 0011
{ 3, 3, 2 },  // 0001 0100
{ 3, 3, 0 },  // 0001 0101
{ 3, 3, 1 },  // 0001 0110
{ 3, 3, 0 },  // 0001 0111
{ 3, 3, 3 },  // 0001 1000
{ 3, 3, 0 },  // 0001 1001
{ 3, 3, 1 },  // 0001 1010
{ 3, 3, 0 },  // 0001 1011
{ 3, 3, 2 },  // 0001 1100
{ 3, 3, 0 },  // 0001 1101
{ 3, 3, 1 },  // 0001 1110
{ 3, 3, 0 },  // 0001 1111
{ 2, 5, 5 },  // 0010 0000
{ 2, 4, 0 },  // 0010 0001
{ 2, 3, 1 },  // 0010 0010
{ 2, 3, 0 },  // 0010 0011
{ 2, 2, 2 },  // 0010 0100
{ 2, 2, 0 },  // 0010 0101
{ 2, 2, 1 },  // 0010 0110
{ 2, 2, 0 },  // 0010 0111
{ 2, 3, 3 },  // 0010 1000
{ 2, 2, 0 },  // 0010 1001
{ 2, 2, 1 },  // 0010 1010
{ 2, 2, 0 },  // 0010 1011
{ 2, 2, 2 },  // 0010 1100
{ 2, 2, 0 },  // 0010 1101
{ 2, 2, 1 },  // 0010 1110
{ 2, 2, 0 },  // 0010 1111
{ 2, 4, 4 },  // 0011 0000
{ 2, 3, 0 },  // 0011 0001
{ 2, 2, 1 },  // 0011 0010
{ 2, 2, 0 },  // 0011 0011
{ 2, 2, 2 },  // 0011 0100
{ 2, 2, 0 },  // 0011 0101
{ 2, 2, 1 },  // 0011 0110
{ 2, 2, 0 },  // 0011 0111
{ 2, 3, 3 },  // 0011 1000
{ 2, 2, 0 },  // 0011 1001
{ 2, 2, 1 },  // 0011 1010
{ 2, 2, 0 },  // 0011 1011
{ 2, 2, 2 },  // 0011 1100
{ 2, 2, 0 },  // 0011 1101
{ 2, 2, 1 },  // 0011 1110
{ 2, 2, 0 },  // 0011 1111
{ 1, 6, 6 },  // 0100 0000
{ 1, 5, 0 },  // 0100 0001
{ 1, 4, 1 },  // 0100 0010
{ 1, 4, 0 },  // 0100 0011
{ 1, 3, 2 },  // 0100 0100
{ 1, 3, 0 },  // 0100 0101
{ 1, 3, 1 },  // 0100 0110
{ 1, 3, 0 },  // 0100 0111
{ 1, 3, 3 },  // 0100 1000
{ 1, 2, 0 },  // 0100 1001
{ 1, 2, 1 },  // 0100 1010
{ 1, 2, 0 },  // 0100 1011
{ 1, 2, 2 },  // 0100 1100
{ 1, 2, 0 },  // 0100 1101
{ 1, 2, 1 },  // 0100 1110
{ 1, 2, 0 },  // 0100 1111
{ 1, 4, 4 },  // 0101 0000
{ 1, 3, 0 },  // 0101 0001
{ 1, 2, 1 },  // 0101 0010
{ 1, 2, 0 },  // 0101 0011
{ 1, 2, 2 },  // 0101 0100
{ 1, 1, 0 },  // 0101 0101
{ 1, 1, 1 },  // 0101 0110
{ 1, 1, 0 },  // 0101 0111
{ 1, 3, 3 },  // 0101 1000
{ 1, 2, 0 },  // 0101 1001
{ 1, 1, 1 },  // 0101 1010
{ 1, 1, 0 },  // 0101 1011
{ 1, 2, 2 },  // 0101 1100
{ 1, 1, 0 },  // 0101 1101
{ 1, 1, 1 },  // 0101 1110
{ 1, 1, 0 },  // 0101 1111
{ 1, 5, 5 },  // 0110 0000
{ 1, 4, 0 },  // 0110 0001
{ 1, 3, 1 },  // 0110 0010
{ 1, 3, 0 },  // 0110 0011
{ 1, 2, 2 },  // 0110 0100
{ 1, 2, 0 },  // 0110 0101
{ 1, 2, 1 },  // 0110 0110
{ 1, 2, 0 },  // 0110 0111
{ 1, 3, 3 },  // 0110 1000
{ 1, 2, 0 },  // 0110 1001
{ 1, 1, 1 },  // 0110 1010
{ 1, 1, 0 },  // 0110 1011
{ 1, 2, 2 },  // 0110 1100
{ 1, 1, 0 },  // 0110 1101
{ 1, 1, 1 },  // 0110 1110
{ 1, 1, 0 },  // 0110 1111
{ 1, 4, 4 },  // 0111 0000
{ 1, 3, 0 },  // 0111 0001
{ 1, 2, 1 },  // 0111 0010
{ 1, 2, 0 },  // 0111 0011
{ 1, 2, 2 },  // 0111 0100
{ 1, 1, 0 },  // 0111 0101
{ 1, 1, 1 },  // 0111 0110
{ 1, 1, 0 },  // 0111 0111
{ 1, 3, 3 },  // 0111 1000
{ 1, 2, 0 },  // 0111 1001
{ 1, 1, 1 },  // 0111 1010
{ 1, 1, 0 },  // 0111 1011
{ 1, 2, 2 },  // 0111 1100
{ 1, 1, 0 },  // 0111 1101
{ 1, 1, 1 },  // 0111 1110
{ 1, 1, 0 },  // 0111 1111
{ 0, 7, 7 },  // 1000 0000
{ 0, 6, 0 },  // 1000 0001
{ 0, 5, 1 },  // 1000 0010
{ 0, 5, 0 },  // 1000 0011
{ 0, 4, 2 },  // 1000 0100
{ 0, 4, 0 },  // 1000 0101
{ 0, 4, 1 },  // 1000 0110
{ 0, 4, 0 },  // 1000 0111
{ 0, 3, 3 },  // 1000 1000
{ 0, 3, 0 },  // 1000 1001
{ 0, 3, 1 },  // 1000 1010
{ 0, 3, 0 },  // 1000 1011
{ 0, 3, 2 },  // 1000 1100
{ 0, 3, 0 },  // 1000 1101
{ 0, 3, 1 },  // 1000 1110
{ 0, 3, 0 },  // 1000 1111
{ 0, 4, 4 },  // 1001 0000
{ 0, 3, 0 },  // 1001 0001
{ 0, 2, 1 },  // 1001 0010
{ 0, 2, 0 },  // 1001 0011
{ 0, 2, 2 },  // 1001 0100
{ 0, 2, 0 },  // 1001 0101
{ 0, 2, 1 },  // 1001 0110
{ 0, 2, 0 },  // 1001 0111
{ 0, 3, 3 },  // 1001 1000
{ 0, 2, 0 },  // 1001 1001
{ 0, 2, 1 },  // 1001 1010
{ 0, 2, 0 },  // 1001 1011
{ 0, 2, 2 },  // 1001 1100
{ 0, 2, 0 },  // 1001 1101
{ 0, 2, 1 },  // 1001 1110
{ 0, 2, 0 },  // 1001 1111
{ 0, 5, 5 },  // 1010 0000
{ 0, 4, 0 },  // 1010 0001
{ 0, 3, 1 },  // 1010 0010
{ 0, 3, 0 },  // 1010 0011
{ 0, 2, 2 },  // 1010 0100
{ 0, 2, 0 },  // 1010 0101
{ 0, 2, 1 },  // 1010 0110
{ 0, 2, 0 },  // 1010 0111
{ 0, 3, 3 },  // 1010 1000
{ 0, 2, 0 },  // 1010 1001
{ 0, 1, 1 },  // 1010 1010
{ 0, 1, 0 },  // 1010 1011
{ 0, 2, 2 },  // 1010 1100
{ 0, 1, 0 },  // 1010 1101
{ 0, 1, 1 },  // 1010 1110
{ 0, 1, 0 },  // 1010 1111
{ 0, 4, 4 },  // 1011 0000
{ 0, 3, 0 },  // 1011 0001
{ 0, 2, 1 },  // 1011 0010
{ 0, 2, 0 },  // 1011 0011
{ 0, 2, 2 },  // 1011 0100
{ 0, 1, 0 },  // 1011 0101
{ 0, 1, 1 },  // 1011 0110
{ 0, 1, 0 },  // 1011 0111
{ 0, 3, 3 },  // 1011 1000
{ 0, 2, 0 },  // 1011 1001
{ 0, 1, 1 },  // 1011 1010
{ 0, 1, 0 },  // 1011 1011
{ 0, 2, 2 },  // 1011 1100
{ 0, 1, 0 },  // 1011 1101
{ 0, 1, 1 },  // 1011 1110
{ 0, 1, 0 },  // 1011 1111
{ 0, 6, 6 },  // 1100 0000
{ 0, 5, 0 },  // 1100 0001
{ 0, 4, 1 },  // 1100 0010
{ 0, 4, 0 },  // 1100 0011
{ 0, 3, 2 },  // 1100 0100
{ 0, 3, 0 },  // 1100 0101
{ 0, 3, 1 },  // 1100 0110
{ 0, 3, 0 },  // 1100 0111
{ 0, 3, 3 },  // 1100 1000
{ 0, 2, 0 },  // 1100 1001
{ 0, 2, 1 },  // 1100 1010
{ 0, 2, 0 },  // 1100 1011
{ 0, 2, 2 },  // 1100 1100
{ 0, 2, 0 },  // 1100 1101
{ 0, 2, 1 },  // 1100 1110
{ 0, 2, 0 },  // 1100 1111
{ 0, 4, 4 },  // 1101 0000
{ 0, 3, 0 },  // 1101 0001
{ 0, 2, 1 },  // 1101 0010
{ 0, 2, 0 },  // 1101 0011
{ 0, 2, 2 },  // 1101 0100
{ 0, 1, 0 },  // 1101 0101
{ 0, 1, 1 },  // 1101 0110
{ 0, 1, 0 },  // 1101 0111
{ 0, 3, 3 },  // 1101 1000
{ 0, 2, 0 },  // 1101 1001
{ 0, 1, 1 },  // 1101 1010
{ 0, 1, 0 },  // 1101 1011
{ 0, 2, 2 },  // 1101 1100
{ 0, 1, 0 },  // 1101 1101
{ 0, 1, 1 },  // 1101 1110
{ 0, 1, 0 },  // 1101 1111
{ 0, 5, 5 },  // 1110 0000
{ 0, 4, 0 },  // 1110 0001
{ 0, 3, 1 },  // 1110 0010
{ 0, 3, 0 },  // 1110 0011
{ 0, 2, 2 },  // 1110 0100
{ 0, 2, 0 },  // 1110 0101
{ 0, 2, 1 },  // 1110 0110
{ 0, 2, 0 },  // 1110 0111
{ 0, 3, 3 },  // 1110 1000
{ 0, 2, 0 },  // 1110 1001
{ 0, 1, 1 },  // 1110 1010
{ 0, 1, 0 },  // 1110 1011
{ 0, 2, 2 },  // 1110 1100
{ 0, 1, 0 },  // 1110 1101
{ 0, 1, 1 },  // 1110 1110
{ 0, 1, 0 },  // 1110 1111
{ 0, 4, 4 },  // 1111 0000
{ 0, 3, 0 },  // 1111 0001
{ 0, 2, 1 },  // 1111 0010
{ 0, 2, 0 },  // 1111 0011
{ 0, 2, 2 },  // 1111 0100
{ 0, 1, 0 },  // 1111 0101
{ 0, 1, 1 },  // 1111 0110
{ 0, 1, 0 },  // 1111 0111
{ 0, 3, 3 },  // 1111 1000
{ 0, 2, 0 },  // 1111 1001
{ 0, 1, 1 },  // 1111 1010
{ 0, 1, 0 },  // 1111 1011
{ 0, 2, 2 },  // 1111 1100
{ 0, 1, 0 },  // 1111 1101
{ 0, 1, 1 },  // 1111 1110
{ 0, 0, 0 },  // 1111 1111
};

逐字遍历数组(32 位或 64 位取决于您的架构)。使用__builtin_clz__builtin_ctz计算每个单词的前导零和尾随零。

连续零有两种情况:

  • 一言以蔽之
  • 跨形容词。

第一种情况很容易检查。第二种情况需要检查此项目的前导零 + 前一项的尾随零是否为>= 6。

注意这些算术技巧:

// remove the trailing one bits
y = x & (x + 1);
x       11100011
+      1
--------
x+1     11100100
x&(x+1) 11100000
// remove the trailing zero bits
z = y | (y - 1);
y       11100000
-      1
--------
y-1     11011111
y|(y-1) 11111111
// isolate the hole
hole = z - x;
hole = z ^ x;
z       11111111
x       11100011
--------
z^x     00011100
// Now you can count the set bits of the hole.
length = bitcount(hole);
// Or you can make it by computing highbit only (log2).
length = highbit(z^y) - highbit(x^y);

因此,一种可能的算法是将这些技巧与大整数算法一起使用并循环直到 length==0(不再有孔)或 length>=n(你从 x=z; 开始下一个循环)。

您可以自己模拟大整数,逐个字节地作用于集合字节,并在没有更多进位时停止。

  • X+1 仅在字节==0xFF
  • y-1 仅在字节==0x00
  • 高位易于在单个字节上编程

这将给出如下所示的内容:

// return 1-based position of highest bit set in a byte
int highbit(unsigned char c)
{
unsigned char n;
int position = 0;
n = c >> 4;
if( n > 0 ) { c=n; position+=4; };
n = c >> 2;
if( n > 0 ) { c=n; position+=2; };
n = c >> 1;
if( n > 0 ) { c=n; position+=1; };
position += c;
return position;
}
int find_consecutive_zeros( unsigned char *bits , int nbytes , int nzero )
{
int i,nTrailingOnes,nTrailingZero,sizeOfNextHole;
unsigned char x,y;
for(i=0 , x=bits[0]; 1; )
{
// perform y=x&(x+1) to count and remove trailing ones
for(;x==0xFF && i<nbytes-1;x=bits[++i]);
y = x&(x+1);
nTrailingOnes = 8*i + highbit( x^y );
// perform x=y|(y-1); to count and remove trailing zeros
for(;y==0 && i<nbytes-1;y=bits[++i]);
x = y|(y-1);
nTrailingZero = 8*i + highbit( x^y );
sizeOfNextHole = nTrailingZero - nTrailingOnes;
// if size of hole is long enough, return it's low bit rank (0-based)
if( sizeOfNextHole >= nzero ) return nTrailingOnes;
// or return -1 if no more hole
if( sizeOfNextHole == 0 ) return -1;
}
}

您可以通过对基集合使用比字节更长的字节来提高效率...

编辑:当您的 nzero 大小固定时加速,例如 6
上面的算法会迭代所有孔,并且可能会在小孔上浪费时间。
您可以通过填充小孔的预先计算表来避免这种情况。

例如,10010101有 3 个太短的孔,因此您可以将其替换为 11111111。
但是您必须保持前导和尾随零不变。

要初始化这样的表,您只需使用 0xFF 和 xor,使用包含 1 位的掩码代替尾随零(x|(x-1))^x,用包含 1 位的掩码代替前导零((1<<highbit(x))-1)^0xFF
为 10000001 添加一个例外,即 1 之间包含 6 个零的唯一字节。

EDIT2: 我首先用最小有效位处理了位序列,这非常适合算术方法。该问题明确地首先使用第一个字节的最高有效位。所以我必须反转位以适应问题,这是在初始化表时完成的......

int reversebits(unsigned char c)
{
c = ((c & 0x0F) << 4) | ((c & 0xF0) >> 4);
c = ((c & 0x33) << 2) | ((c & 0xCC) >> 2);
c = ((c & 0x55) << 1) | ((c & 0xAA) >> 1);
return c;
}
void initializeFillShortHoles(unsigned char fillShortHoles[256])
for(unsigned int x=0;x<256;x++) {
fillShortHoles[reversebits(x)] = ((1<<highbit(x))-1) ^ (x|(x-1)) ^ x;
}
fillShortHoles[0]=0;     // no use to reverse bits for those two
fillShortHoles[129]=129;
}

然后,您只需将出现的x=bits[ i ]替换为x=fillShortHoles[ bits[ i ] ]

即可完成:
int find_6_consecutive_zeros( unsigned char *bits , int nbytes )
{
static unsigned char fillShortHoles[256];
static unsigned char highbitTable[256];
static first=1;
int i,nTrailingOnes,nTrailingZero,sizeOfNextHole;
unsigned char x,y;
if (first)
{
first = 0;
initializeFillShortHoles( fillShortHoles );
for(i=0;i<256;i++) highbitTable[i]=highbit(i);
}
for(x=fillShortHoles[bits[i=0]]; 1; )
{
// perform y=x&(x+1) to count trailing ones
for(;x==0xFF && i<nbytes-1;x=fillShortHoles[bits[++i]]);
y = x&(x+1);
nTrailingOnes = 8*i + highbitTable[ x^y ];
// perform z=y|(y-1); to count trailing zeros
for(;y==0 && i<nbytes-1;y=fillShortHoles[bits[++i]]);
x = y|(y-1);
nTrailingZero = 8*i + highbitTable[ x^y ];
sizeOfNextHole = nTrailingZero - nTrailingOnes;
// if size of hole is long enough, return it's low bit rank (0-based)
if( sizeOfNextHole >= 6 ) return nTrailingOnes;
// or return -1 if no more hole
if( sizeOfNextHole == 0 ) return -1;
}
}

EDIT3:最后,对于 nzero<=9,更快的方法是缓存每对字节的位置。

int find_n_consecutive_zeros_bypair( unsigned char *bits , int nbytes , int nzero)
{
static int first=1;
static signed char holepositionbypair[8][65536];
signed char position;
int i;
unsigned short x;
if (first)
{
first = 0;
for(i=0;i<8;i++) {
initializeHolePositionByPair( holepositionbypair[i] , i+1 );
}
}
for (i=0 , x=0xFF; i<nbytes; i++) {
x = (x << 8) + bits[i];
if( (position = holepositionbypair[nzero-1][x]) >= 0) return (i-1)*8 + position;
}
return -1;
}

请注意,初始化 x=0xFF 将处理 nbytes<2 的情况。

无论你如何填充缓存洞位置对,它只会在初始化时被调用。我当然提出算术方式:

int highbit16(unsigned short c)
{
unsigned short n;
int position = 0;
n = c >> 8;
if( n ) { c=n; position+=8; };
n = c >> 4;
if( n ) { c=n; position+=4; };
n = c >> 2;
if( n ) { c=n; position+=2; };
n = c >> 1;
if( n ) { c=n; position+=1; };
position += c;
return position;
}
unsigned short reversebits16(unsigned short c)
{
c = ((c & 0x00FF) << 8) | ((c & 0xFF00) >> 8);
c = ((c & 0x0F0F) << 4) | ((c & 0xF0F0) >> 4);
c = ((c & 0x3333) << 2) | ((c & 0xCCCC) >> 2);
c = ((c & 0x5555) << 1) | ((c & 0xAAAA) >> 1);
return c;
}
initializeHolePositionByPair(signed char holepositionbypair[65536],int n)
{
int i,n1,n0;
unsigned short x,y;
signed char position;
for(i=0;i<65536;i++) {
position = -1;
x = i;
while(x != 0xFFFF) {
/* remove trailing ones */
y = x&(x+1);
n1 = highbit16(x^y);
/* remove trailing zeros */
x = y|(y-1);
n0 = highbit16(x^y);
if(n0-n1>=n) {
position = n1; break;
}
}
holepositionbypair[reversebits16(i)] = position;
}
}

在这里,尝试此代码。

int dataLen = 500;
char data[dataLen];
//Get the data in place with whatever is your algorithm.
int i,j;
unsigned int dataSample;
unsigned int mask;
for(i=0;i<dataLen-1;i++){
dataSample=((unsigned int)(data[i+1]) << 8) | (unsigned int) (data[i]);
mask=(1<<6) - 1 ; //6 consecutive 1's
for(j=0;j<8;j++){
if((dataSample & (mask << j)) == 0){
printf("Found consecutive 6 zeros at byte %d offset %dn",i,j);
j+=5; // Followed by j++ makes it j+=6.
}
}
}

对于一个字节,你只需要做3个测试:

if( (byte & 0x3F) == 0) { /* Found */ }
if( (byte & 0x7E) == 0) { /* Found */ }
if( (byte & 0xFC) == 0) { /* Found */ }

将其扩展到更广泛的值应该相对容易。例如,对于uint32_t

tempValue1 = value & 0x3F3F3F3F;
tempValue2 = value & 0x7E7E7E7E;
tempValue3 = value & 0xFCFCFCFC;
if( ( (uint8_t)tempValue1 == 0) { /* Found */ }
if( ( (uint8_t)tempValue2 == 0) { /* Found */ }
if( ( (uint8_t)tempValue3 == 0) { /* Found */ }
tempValue1 >>= 8;
tempValue2 >>= 8;
tempValue3 >>= 8;
if( ( (uint8_t)tempValue1 == 0) { /* Found */ }
if( ( (uint8_t)tempValue2 == 0) { /* Found */ }
if( ( (uint8_t)tempValue3 == 0) { /* Found */ }
tempValue1 >>= 8;
tempValue2 >>= 8;
tempValue3 >>= 8;
if( ( (uint8_t)tempValue1 == 0) { /* Found */ }
if( ( (uint8_t)tempValue2 == 0) { /* Found */ }
if( ( (uint8_t)tempValue3 == 0) { /* Found */ }
tempValue1 >>= 8;
tempValue2 >>= 8;
tempValue3 >>= 8;
if( ( (uint8_t)tempValue1 == 0) { /* Found */ }
if( ( (uint8_t)tempValue2 == 0) { /* Found */ }
if( ( (uint8_t)tempValue3 == 0) { /* Found */ }

上面的代码看起来不是最好的方法(因为它可能不是)。我故意这样写它,以便更容易看到如何使用 SSE 完成它。对于SSE,它将类似于上述(但更广泛)。

但是,对于SSE,您可以并行进行所有比较并摆脱大量分支。例如,您可以 AND 使用 3 个掩码中的每一个并PCMPEQB3 次,然后一起 OR 这些结果,然后执行一个PMOVMSKB;这将为您提供一个 16 位值(代表 16 个字节 - 每个源字节一位),可以用单个if(result_flags == 0) { /* None of the 16 bytes matched */ }进行测试,其中最终测试可能在"do/while"循环的末尾。

假设您正在寻找 6 个连续的零。您可以使用这样的代码片段:

unsigned d1 = 0xff;
for (int i = 0; i < dataLen; i += 3) {
unsigned d1 = (d1 << 24) | (data[i] << 16) | (data [i+1] << 8) | (data[i+2]);
unsigned d2 = ~d1; // ones
unsigned d3 = d2 & (d2 << 2) & (d2 << 4);
unsigned d4 = d3 & (d3 << 1);
if (!d4) continue;
doSomethingWithTheSequence(i, d4);
}

d1会将上一次运行的最后一个字节与三个新字节合并在一起。因此,您可以以 3 的倍数迭代数据。您可以尝试 2 的倍数,这可能更有效,因为您可以将数据视为 16 位原子量。您也可以尝试 4 的倍数,并对 64 位数字进行后续计算,尤其是在 64 位平台上。或者,您引入了跨越字节的零序列的特殊情况。

d2反转位模式,这很有用,因为移位会引入人工零,但绝不会引入人工零。d3在偏移量 0、2 和 4 处查找三个匹配的。 然后d4添加另一位偏移量,从而合并从 0 到 5 的所有偏移量。因此,当且仅当d4d1中连续运行 6 个零时,它将不为零。然后你可以使用__builtin_clz来识别d4中最重要的位置,这也是这6位中第一个在d1中的位置。从中您可以获得data的位置。

您可以使代码适应其他运行长度,方法是添加一个循环并希望编译器对其进行优化,或者通过提供内联模板函数,该函数以适合所需运行长度的方式从d2计算d4

让我试试 - 怎么样:

class bitIterator{
unsigned char* farray;
int foffset;
public:
bitIterator(unsigned char* array, int offset)
:farray(array), foffset(offset)
{}
bool operator*() const {
return (farray[foffset/8] >> (7 - foffset%8)) & 1;
}
bitIterator& operator++(){
foffset++;
return *this;
}
int offset() const {
return foffset; 
}
};
// Just to demonstrate how to call it
int find_first_n(unsigned char* buffer, int length, int n){
return std::search_n(bitIterator(buffer, 0), bitIterator(buffer, length*8), n, 0).offset();
}

那只是为了好玩...

现在,如果你真的想从中挤出一些性能,我会建议

int find_first_n(unsigned char* buffer, int length, int n){
int prev_trail = 0;
unsigned int* buf = reinterpret_cast<unsigned int*>(buffer);
int len = length/sizeof(int); 
// Last few bytes will need special treatment if your array is not multple of int-size length.
// However last bytes should not influence overall performance of code - assuming it is used on rather long arrays.
// If you plan using rather short arrays (20-50 bytes?) you will probably be better off just using plain char.
for (int i=0; i<len; i++){
if (!buf[i]) return i*sizeof(int)*8-prev_trail; // __builtin_clz and __builtin_ctz are undefined on zero ;
// EDIT:
if (!~buf[i]){
prev_trail = 0;
continue;
}
// END EDIT!

int shft = __builtin_clz(buf[i]);
if (shft + prev_trail >= n) return i*sizeof(int)*8-prev_trail; // Leading + previous trailing <= n
// Not enough leading zeros, continue search.
prev_trail = __builtin_ctz(buf[i]); // Store it to use for next int in array
unsigned int tmp =0;               
while(shft < sizeof(int)*8-prev_trail){   // While we haven't got to trailing zeros in this uint
tmp = buf[i] << shft;               // Shift-out leading zeros;
shft += (__builtin_clz(~tmp));
tmp = buf[i] << shft;               // Shift-out leading ones;
lead_zeros = __builtin_clz(tmp);
if (lead_zeros >= n)                // and see if have enough leading zeros.
return i*sizeof(int)*8+shft;
shft += lead_zeros;
}
return length*8; // Not found :(
}

这很难阅读,但概念很简单 - 只需迭代每个 int 大小的块,并查看每个块的前导零 + 前一个块的尾随零是否>= n。如果不是重复移出前导零和后导零(设置位),并再次检查尾随零>= n,只要我们没有尾随字节。

还有一个想法:

int find_first_n(unsigned char* buffer, int length, int n){
union {
unsigned char chr[2];
unsigned int uit;
};
unsigned int onemask = 0x8000;
for (int i = 1 ; i<n; i++) onemask = onemask | (onemask >> 1);
int* masks = new int[8];
for (int i = 0; i<8; i++){
masks[i]=onemask >> i;
}
// generating masks - eg. for n == 3 would be:
// [ 1110 0000 0000 0000 ]
// [ 0111 0000 0000 0000 ]
// [ 0011 1000 0000 0000 ]
// [ 0001 1100 0000 0000 ]
// [ ... ]
// [ 0000 0001 1100 0000 ]

uit = 0;
for (int i=0; i<length-1; i++){
chr[0] = buffer[i];
chr[1] = buffer[i+1];
for (int j=0; j<8; j++){
if (!(uit & masks[j])) return i*8+j;
}
}
chr[0] = buffer[length-1];
chr[1] = 0xff; // Fill with ones at the end.
for (int j=0; j<8; j++){
if (!(uit & masks[j])) return (length-1)*8+j;
}
return length*8; // Not found :(
}

您可能想将指向仍在缓冲区中的变量的指针直接投射到 word (reinterpret_cast<unsigned short int*>(buffer[i])),并在不使用联合的情况下应用掩码。但请注意,此类操作中有一半会使用未对齐的变量,这可能会降低性能,并且在某些平台上甚至会生成异常。

我会通过利用 x86 小字节序和未对齐的内存访问功能来解决这个问题。 使用 popcount 查找候选单词,然后使用 __builtin_ctz 通过循环查找位置。这消除了代码中的一个循环,并在单词不是候选词时避免位搜索循环。 在大端计算机上,您需要使用 htons(*(无符号短 *)p) 来交换字节。 这需要一台允许未对齐的单词访问的计算机。 您还需要在数组末尾额外0xFF字节上设置一个栅栏。

unsigned char *p;
unsigned short v;
int c;
for (p = data; p < data + sizeof(data); p++)
if (__builtin_popcount(*(unsigned short *)p) <= 16 - desired)
{ for (v = *(unsigned short *)p, c = 0; __builtin_ctz(v) < desired; v>>=1, c++) ;
if (c < 16) // Gotcha @ p starting at bit c
break;
}
if (c == 16 || p >= data + sizeof(data)) // didn't find
else // found