如何将位转换为字节?

Dav*_*own 9 c# byte bit-manipulation

我有一个128个布尔数组,代表位.如何将这128位表示转换为16字节?

例:

我有一个看起来像这样的数组:

0110001100110000100010111011001011010011010001010001101101001100
1000010000000000001000111111111101000011111001111011111011111001
Run Code Online (Sandbox Code Playgroud)

(转换为1和0更简洁)

我需要将这些位转换为以下字节数组:

99 48 139 178 211 69 27 76 132 0 35 255 67 231 190 249
Run Code Online (Sandbox Code Playgroud)

编辑:这似乎不起作用:

public byte[] ToByteArray() {
    int numBytes = Count / 8;

    if (_bits.Count % 8 != 0) numBytes++;

    byte[] bytes = new byte[numBytes];

    int byteIndex = 0, bitIndex = 0;

    for (int i = 0; i < _bits.Count; i++) {
        if (_bits[i])
            bytes[byteIndex] |= (byte)(1 << bitIndex);

        bitIndex++;
        if (bitIndex == 8) {
            bitIndex = 0;
            byteIndex++;
        }
    }

    return bytes;
}
Run Code Online (Sandbox Code Playgroud)

它输出:

198 12 209 77 203 162 216 50 33 0 196 255 194 231 125 159
Run Code Online (Sandbox Code Playgroud)

CAd*_*ker 10

代码将第一位视为字的低位,因此您最终会将每个字反转.作为一个快速和肮脏的解决方案,试试这个:

bytes[byteIndex] |= (byte)(1 << (7-bitIndex));
Run Code Online (Sandbox Code Playgroud)

这会将数组中的第一位置于第一个字节中的最高位置,等等.


Bri*_*ndy 5

我不知道是否有自动方法,但你可以用一个简单的算法来做.

简单的算法:

  1. 创建一个将用作输出缓冲区的字节数组,并将所有字节初始化为0.此数组的大小应基于输入布尔数组的长度:ceil(bool_array_length/8.0)

  2. 声明一个索引变量用作当前字节,并将其设置为0.这将索引保存在输出缓冲区中.

  3. 迭代输入布尔数组中的每个元素.
    3.1.左位将数字1移位数组索引mod 8.将此数字称为掩码.
    3.2.将您的字节索引计算为数组div 8中的当前索引
    .如果true输入布尔数组中的当前索引具有布尔值,则bitwise OR使用当前字节和掩码执行a .


Luc*_*cas 5

bool[] bools = ...
BitArray a = new BitArray(bools);
byte[] bytes = new byte[a.Length / 8];
a.CopyTo(bytes, 0);
Run Code Online (Sandbox Code Playgroud)

编辑:实际上这也返回:

198 12 209 77 203 162 216 50 33 0 196 255 194 231 125 159
Run Code Online (Sandbox Code Playgroud)

字节顺序错误?无论如何,我都会留下答案,以供参考。


编辑:您可以通过反转数组来使用BitArray.CopyTo(),如下所示:

bool[] bools = ...
Array.Reverse(bools); // NOTE: this modifies your original array
BitArray a = new BitArray(bools);
byte[] bytes = new byte[a.Length / 8];
a.CopyTo(bytes, 0);
Array.Reverse(bytes);
Run Code Online (Sandbox Code Playgroud)