- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
有一个关于 reading a C/C++ data structure in C# from a byte array 的富有洞察力的问题,但我无法让代码适用于我的大端(网络字节顺序)字节集合。 (编辑:请注意,我的真实结构不止一个字段。)有没有一种方法可以将字节编码到结构的大端版本中,然后以框架的字节顺序(主机的字节顺序)提取值,通常是小尾数)?
(请注意,反转字节数组不起作用 - 每个值的字节都必须反转,这不会为您提供与反转所有字节相同的集合。)
这应该总结了我正在寻找的内容(LE=LittleEndian,BE=BigEndian):
void Main()
{
var leBytes = new byte[] {1, 0, 2, 0};
var beBytes = new byte[] {0, 1, 0, 2};
Foo fooLe = ByteArrayToStructure<Foo>(leBytes);
Foo fooBe = ByteArrayToStructureBigEndian<Foo>(beBytes);
Assert.AreEqual(fooLe, fooBe);
}
[StructLayout(LayoutKind.Explicit, Size=4)]
public struct Foo {
[FieldOffset(0)]
public ushort firstUshort;
[FieldOffset(2)]
public ushort secondUshort;
}
T ByteArrayToStructure<T>(byte[] bytes) where T: struct
{
GCHandle handle = GCHandle.Alloc(bytes, GCHandleType.Pinned);
T stuff = (T)Marshal.PtrToStructure(handle.AddrOfPinnedObject(),typeof(T));
handle.Free();
return stuff;
}
T ByteArrayToStructureBigEndian<T>(byte[] bytes) where T: struct
{
???
}
其他有用的链接:
Byte of a struct and onto endian concerns
A little more on bytes and endianness (byte order)
Read binary files more efficiently using C#
最佳答案
这是交换字节顺序的另一种解决方案。
它是根据 Adam Robinsons 解决方案进行调整的:https://stackoverflow.com/a/2624377/1254743
它甚至能够处理嵌套结构。
public static class FooTest
{
[StructLayout(LayoutKind.Sequential, Pack = 1)]
public struct Foo2
{
public byte b1;
public short s;
public ushort S;
public int i;
public uint I;
public long l;
public ulong L;
public float f;
public double d;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 10)]
public string MyString;
}
[StructLayout(LayoutKind.Sequential, Pack = 1)]
public struct Foo
{
public byte b1;
public short s;
public ushort S;
public int i;
public uint I;
public long l;
public ulong L;
public float f;
public double d;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 10)]
public string MyString;
public Foo2 foo2;
}
public static void test()
{
Foo2 sample2 = new Foo2()
{
b1 = 0x01,
s = 0x0203,
S = 0x0405,
i = 0x06070809,
I = 0x0a0b0c0d,
l = 0xe0f101112131415,
L = 0x161718191a1b1c,
f = 1.234f,
d = 4.56789,
MyString = @"123456789", // null terminated => only 9 characters!
};
Foo sample = new Foo()
{
b1 = 0x01,
s = 0x0203,
S = 0x0405,
i = 0x06070809,
I = 0x0a0b0c0d,
l = 0xe0f101112131415,
L = 0x161718191a1b1c,
f = 1.234f,
d = 4.56789,
MyString = @"123456789", // null terminated => only 9 characters!
foo2 = sample2,
};
var bytes_LE = Dummy.StructToBytes(sample, Endianness.LittleEndian);
var restoredLEAsLE = Dummy.BytesToStruct<Foo>(bytes_LE, Endianness.LittleEndian);
var restoredLEAsBE = Dummy.BytesToStruct<Foo>(bytes_LE, Endianness.BigEndian);
var bytes_BE = Dummy.StructToBytes(sample, Endianness.BigEndian);
var restoredBEAsLE = Dummy.BytesToStruct<Foo>(bytes_BE, Endianness.LittleEndian);
var restoredBEAsBE = Dummy.BytesToStruct<Foo>(bytes_BE, Endianness.BigEndian);
Debug.Assert(sample.Equals(restoredLEAsLE));
Debug.Assert(sample.Equals(restoredBEAsBE));
Debug.Assert(restoredBEAsLE.Equals(restoredLEAsBE));
}
public enum Endianness
{
BigEndian,
LittleEndian
}
private static void MaybeAdjustEndianness(Type type, byte[] data, Endianness endianness, int startOffset = 0)
{
if ((BitConverter.IsLittleEndian) == (endianness == Endianness.LittleEndian))
{
// nothing to change => return
return;
}
foreach (var field in type.GetFields())
{
var fieldType = field.FieldType;
if (field.IsStatic)
// don't process static fields
continue;
if (fieldType == typeof(string))
// don't swap bytes for strings
continue;
var offset = Marshal.OffsetOf(type, field.Name).ToInt32();
// handle enums
if (fieldType.IsEnum)
fieldType = Enum.GetUnderlyingType(fieldType);
// check for sub-fields to recurse if necessary
var subFields = fieldType.GetFields().Where(subField => subField.IsStatic == false).ToArray();
var effectiveOffset = startOffset + offset;
if (subFields.Length == 0)
{
Array.Reverse(data, effectiveOffset, Marshal.SizeOf(fieldType));
}
else
{
// recurse
MaybeAdjustEndianness(fieldType, data, endianness, effectiveOffset);
}
}
}
internal static T BytesToStruct<T>(byte[] rawData, Endianness endianness) where T : struct
{
T result = default(T);
MaybeAdjustEndianness(typeof(T), rawData, endianness);
GCHandle handle = GCHandle.Alloc(rawData, GCHandleType.Pinned);
try
{
IntPtr rawDataPtr = handle.AddrOfPinnedObject();
result = (T)Marshal.PtrToStructure(rawDataPtr, typeof(T));
}
finally
{
handle.Free();
}
return result;
}
internal static byte[] StructToBytes<T>(T data, Endianness endianness) where T : struct
{
byte[] rawData = new byte[Marshal.SizeOf(data)];
GCHandle handle = GCHandle.Alloc(rawData, GCHandleType.Pinned);
try
{
IntPtr rawDataPtr = handle.AddrOfPinnedObject();
Marshal.StructureToPtr(data, rawDataPtr, false);
}
finally
{
handle.Free();
}
MaybeAdjustEndianness(typeof(T), rawData, endianness);
return rawData;
}
}
关于.net - 将大端字节集合编码到结构中以提取值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2480116/
我有一条记录: Tconnecting = record var a: int64; b: integer; c: integer; end; 我需要使用UDP协议
我知道,我知道,这听起来很简单,但我很困惑。如果您有以下输入 Data : 1100 0101 1010 1101 并且您想将大端转换为小端,或反之,您认为以下哪项是正确的输出? 1011 0101
我目前正在处理的项目需要与我们不制作的客户端系统交互,因此我们无法控制数据的发送方式。问题是在 C# 中工作,它似乎对 UCS-2 没有任何支持,对 big-endian 的支持也很少。 (据我所知)
我正在编写一个函数,将 ascii 字符串转换为其大端字节序的二进制表示形式。 这是我的代码: int count = 0; for (int i = 0; i = 0; k--) {
我从 MSDN 中找到了以下函数它将 unsigned long 从网络字节转换为主机字节顺序的 unsigned long,即在 little-endian 中定义为: u_long WSAAPI
我正在使用 AES Rijndael 的公共(public)领域引用实现,通常以名称“rijndael-fst-3.0.zip”分发。我打算用它来加密网络数据,我想知道加密结果在大/小端架构上是否会有
我有一个十六进制字符串,其内容为18000000,该字符串采用主机字节顺序(小端),我需要将其转换为网络字节顺序(大端)。生成的十六进制字符串将为 00000018。 总而言之,我需要转换180000
我有一个十六进制字符串,其内容为18000000,该字符串采用主机字节顺序(小端),我需要将其转换为网络字节顺序(大端)。生成的十六进制字符串将为 00000018。 总而言之,我需要转换180000
我正在生成用于模拟网络的单元测试的二进制文件。通过网络发送的一些数据采用 Little Endian 或 Big Endian 模式,我想使用 stub 模拟这些数据并创建二进制文件。 因此,换句话说
这个问题在这里已经有了答案: Endianness Work-around Needed (2 个答案) 关闭 9 年前。 我有一个包含 n 个单精度值的二进制文件。我知道写入数据时使用的格式是大端
我有一个 4 字节编码的日期(大端),我试图从二进制文件中读取它。 我愿意: char date[4]; long seconds; s = read(fd, date, sizeof(char) *
在 Python2.7 中,通过 USB 批量传输,我从相机中获取图像帧: frame = dev.read(0x81, 0x2B6B0, 1000) 我知道一帧是 342x260 = 88920 像
我有一个 byte[4],它包含一个 32 位无符号整数(按大端顺序),我需要将它转换为 long(因为 int 不能保存无符号数)。 此外,我该如何反之(即从包含 32 位无符号整数的 long 到
public static void main(String[] args) { File inFile = null; if (0 0) { getMarker(re
我正在使用 Node.JS。 Node's buffers支持小端 UCS-2,但不支持我需要的大端。我该怎么做? 最佳答案 根据维基百科,UCS-2 should always be big-end
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 7 年前。 Improve this ques
我需要通过串行线路 (RS-232) 从 Arduino 接收十六进制编码的单精度大端浮点值。如何将它们转换为 Python 的 double 大端字节序的 float ? Arduino 发送类似“
类似于如何使用包: N An unsigned long (32-bit) in "network" (big-endian) order. 是否有任何方法可以在 Perl 中以“网络”(大端)顺序
我有一个这样的 C 结构...... struct icmp_prefixopt { u_int8_t icmpopt_type; u_int8_t icm
我正在使用 PHP、fopen、fseek、pack 等创建然后将数据写入文件(一个新的“ESRI Shape 文件”)。文件规范在这里 http://www.esri.com/library/whi
我是一名优秀的程序员,十分优秀!