[英]Convert a List of objects to a byte array in c#
我有这个结构定义:
public struct ChangedByte
{
public byte R;
public byte G;
public byte B;
public int x;
public int y;
}
我创建了一个:
List<ChangedByte> testme = new List<ChangedByte>();
(并添加了项目)
我将其转换为数组:
ChangedByte[] alpha = testme.ToArray();
我有这个功能,我在SO上发现了类似的问题:
byte[] StructureToByteArray(object obj)
{
int len = Marshal.SizeOf(obj);
byte[] arr = new byte[len];
IntPtr ptr = Marshal.AllocHGlobal(len);
Marshal.StructureToPtr(obj, ptr, true);
Marshal.Copy(ptr, arr, 0, len);
Marshal.FreeHGlobal(ptr);
return arr;
}
我称之为:
byte[] test = StructureToByteArray(alpha);
我收到错误:
类型'new_encoder.Form1 + ChangedByte []'不能作为非托管结构封送; 不能计算有意义的大小或偏移量。
我想将一个对象列表转换为一个字节数组(并可能避免序列化,因为它往往会增加数组的大小)
可以吗?
额外:
我试过将Struct修改为:
public struct ChangedByte
{
[MarshalAs(UnmanagedType.LPWStr)]
public byte R;
[MarshalAs(UnmanagedType.LPWStr)]
public byte G;
[MarshalAs(UnmanagedType.LPWStr)]
public byte B;
[MarshalAs(UnmanagedType.LPWStr)]
public int x;
[MarshalAs(UnmanagedType.LPWStr)]
public int y;
}
但仍然是同样的错误。
您可以使用以下方法:
static byte[] ArrayToBytes<T>(T[] array) where T : struct
{
IntPtr ptr = IntPtr.Zero;
try
{
int len = Marshal.SizeOf(typeof(T));
int totalLen = array.Length * len;
ptr = Marshal.AllocHGlobal(len);
for (int i = 0; i < array.Length; i++)
{
Marshal.StructureToPtr(array[i], ptr + i * len, false);
}
var bytes = new byte[totalLen];
Marshal.Copy(ptr, bytes, 0, totalLen);
return bytes;
}
finally
{
if (ptr != IntPtr.Zero)
Marshal.FreeHGlobal(ptr);
}
}
static T[] ArrayFromBytes<T>(byte[] bytes) where T : struct
{
IntPtr ptr = IntPtr.Zero;
try
{
int len = Marshal.SizeOf(typeof(T));
int count = bytes.Length / len;
ptr = Marshal.AllocHGlobal(bytes.Length);
Marshal.Copy(bytes, 0, ptr, bytes.Length);
T[] array = new T[count];
for (int i = 0; i < count; i++)
{
array[i] = Marshal.PtrToStructure<T>(ptr + i * len);
}
return array;
}
finally
{
if (ptr != IntPtr.Zero)
Marshal.FreeHGlobal(ptr);
}
}
并像这样使用它们:
byte[] test = ArrayToBytes(alpha); // serialize
ChangedByte[] alpha2 = ArrayFromBytes<ChangedByte>(test); // deserialize
在这种情况下,由于您不仅要转换结构,而且还要转换结构数组,因此需要循环遍历数组并转换每个项目。 使用指针算法在缓冲区中移动并添加项目。
public static byte[] StructureArrayToByteArray(ChangedByte[] objs)
{
int structSize = Marshal.SizeOf(typeof(ChangedByte));
int len = objs.Length * structSize;
byte[] arr = new byte[len];
IntPtr ptr = Marshal.AllocHGlobal(len);
for (int i = 0; i < objs.Length; i++ ) {
Marshal.StructureToPtr(objs[i], ptr + i*structSize, true);
}
Marshal.Copy(ptr, arr, 0, len);
Marshal.FreeHGlobal(ptr);
return arr;
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.