Ich habe eine BitArray
mit der Länge von 8 und brauche eine Funktion, um sie in eine byte
umzuwandeln. Wie es geht?
Insbesondere brauche ich eine korrekte Funktion von ConvertToByte
:
BitArray bit = new BitArray(new bool[]
{
false, false, false, false,
false, false, false, true
});
//How to write ConvertToByte
byte myByte = ConvertToByte(bit);
var recoveredBit = new BitArray(new[] { myByte });
Assert.AreEqual(bit, recoveredBit);
Das sollte funktionieren:
byte ConvertToByte(BitArray bits)
{
if (bits.Count != 8)
{
throw new ArgumentException("bits");
}
byte[] bytes = new byte[1];
bits.CopyTo(bytes, 0);
return bytes[0];
}
Ein bisschen später Beitrag, aber das funktioniert für mich:
public static byte[] BitArrayToByteArray(BitArray bits)
{
byte[] ret = new byte[(bits.Length - 1) / 8 + 1];
bits.CopyTo(ret, 0);
return ret;
}
Arbeitet mit:
string text = "Test";
byte[] bytes = System.Text.Encoding.ASCII.GetBytes(text);
BitArray bits = new BitArray(bytes);
bytes[] bytesBack = BitArrayToByteArray(bits);
string textBack = System.Text.Encoding.ASCII.GetString(bytesBack);
// bytes == bytesBack
// text = textBack
.
Die Lösung eines armen Mannes:
protected byte ConvertToByte(BitArray bits)
{
if (bits.Count != 8)
{
throw new ArgumentException("illegal number of bits");
}
byte b = 0;
if (bits.Get(7)) b++;
if (bits.Get(6)) b += 2;
if (bits.Get(5)) b += 4;
if (bits.Get(4)) b += 8;
if (bits.Get(3)) b += 16;
if (bits.Get(2)) b += 32;
if (bits.Get(1)) b += 64;
if (bits.Get(0)) b += 128;
return b;
}
Das sollte den Trick tun. Die vorige Antwort ist jedoch wahrscheinlich die bessere Option.
public byte ConvertToByte(BitArray bits)
{
if (bits.Count > 8)
throw new ArgumentException("ConvertToByte can only work with a BitArray containing a maximum of 8 values");
byte result = 0;
for (byte i = 0; i < bits.Count; i++)
{
if (bits[i])
result |= (byte)(1 << i);
}
return result;
}
In dem von Ihnen geposteten Beispiel lautet das resultierende Byte 0x80. Mit anderen Worten, der erste Wert im BitArray entspricht dem ersten Bit im zurückgegebenen Byte.
Das sollte das ultimative sein. Funktioniert mit beliebiger Länge des Arrays.
private List<byte> BoolList2ByteList(List<bool> values)
{
List<byte> ret = new List<byte>();
int count = 0;
byte currentByte = 0;
foreach (bool b in values)
{
if (b) currentByte |= (byte)(1 << count);
count++;
if (count == 7) { ret.Add(currentByte); currentByte = 0; count = 0; };
}
if (count < 7) ret.Add(currentByte);
return ret;
}
Leider ist die BitArray-Klasse teilweise in der .Net Core-Klasse (UWP) implementiert. Beispielsweise kann die BitArray-Klasse die Methoden CopyTo () und Count () nicht aufrufen. Ich habe diese Erweiterung geschrieben, um die Lücke zu füllen:
public static IEnumerable<Byte> ToBytes(this BitArray bits, bool MSB = false)
{
int bitCount = 7;
int outByte = 0;
foreach (bool bitValue in bits)
{
if (bitValue)
outByte |= MSB ? 1 << bitCount : 1 << (7 - bitCount);
if (bitCount == 0)
{
yield return (byte) outByte;
bitCount = 8;
outByte = 0;
}
bitCount--;
}
// Last partially decoded byte
if (bitCount < 7)
yield return (byte) outByte;
}
}
Die Methode decodiert das BitArray in ein Byte-Array unter Verwendung der LSB-Logik (Less Significant Byte). Dies ist die gleiche Logik, die von der BitArray-Klasse verwendet wird. Das Aufrufen der Methode, wobei der MSB-Parameter auf true gesetzt ist, erzeugt eine decodierte MSB-Bytefolge. Beachten Sie in diesem Fall, dass Sie möglicherweise auch die endgültige Sammlung von Ausgabebytes umkehren müssen.
Neben der @ JonSkeet-Antwort können Sie die generische Methode als Schlag verwenden:
public static byte ToByte(this BitArray bits)
{
if (bits.Count != 8)
{
throw new ArgumentException("bits");
}
byte[] bytes = new byte[1];
bits.CopyTo(bytes, 0);
return bytes[0];
}
Und verwenden wie:
BitArray foo = new BitArray(new bool[]
{
false, false, false, false,false, false, false, true
});
foo.ToByte();
Little-Endian-Byte-Array-Konverter: Das erste Bit (mit "0" indiziert) im BitArray Wird als das niedrigstwertige Bit (ganz rechts im Bit-Oktett) bezeichnet, das als "Null" oder "Eins" als binär interpretiert wird.
public static class BitArrayExtender {
public static byte[] ToByteArray( this BitArray bits ) {
const int BYTE = 8;
int length = ( bits.Count / BYTE ) + ( (bits.Count % BYTE == 0) ? 0 : 1 );
var bytes = new byte[ length ];
for ( int i = 0; i < bits.Length; i++ ) {
int bitIndex = i % BYTE;
int byteIndex = i / BYTE;
int mask = (bits[ i ] ? 1 : 0) << bitIndex;
bytes[ byteIndex ] |= (byte)mask;
}//for
return bytes;
}//ToByteArray
}//class
byte GetByte(BitArray input)
{
int len = input.Length;
if (len > 8)
len = 8;
int output = 0;
for (int i = 0; i < len; i++)
if (input.Get(i))
output += (1 << (len - 1 - i)); //this part depends on your system (Big/Little)
//output += (1 << i); //depends on system
return (byte)output;
}
Prost!
Ein Beispiel aus meinen Codes:
Public Shared Function BytesXor(a1 As Byte(), a2 As Byte()) As Byte()
Dim ba1 As BitArray = New BitArray(a1)
Dim ba2 As BitArray = New BitArray(a2)
Dim intLength As Integer = System.Math.Min(a1.Length, a2.Length)
Dim RetrunValue(intLength - 1) As Byte
If ba1.Length > ba2.Length Then ba1.Length = ba2.Length
If ba2.Length > ba1.Length Then ba2.Length = ba1.Length
Dim ba3 As BitArray = ba1.Xor(ba2)
Dim p As Integer = 0
For i As Integer = 0 To intLength - 1
Dim v As Integer = 0
For j As Integer = 0 To 7
If ba3.Get(p) Then
'BitArray(Byte()) sorts bits from lower to higher
'"BitArray to Byte" must be put by reverse order
v += 1 << j
End If
p += 1
Next
RetrunValue(i) = CByte(v)
Next
Return RetrunValue
End Function