不同应用程序(WPF 和 Unity)中的 C# struct 序列化返回不同大小的字节数组

Posted

技术标签:

【中文标题】不同应用程序(WPF 和 Unity)中的 C# struct 序列化返回不同大小的字节数组【英文标题】:C# struct serialization in different applications (WPF and Unity) returns byte array of different sizes 【发布时间】:2018-12-26 05:43:56 【问题描述】:

我有一个 dll,它具有结构定义以及对它们进行序列化和反序列化的几个函数。我想将它们放在一个 dll 中,以便可以在不同的应用程序之间共享完全相同的结构。

该 dll 由 WPF 应用程序(通过 MQTT 发送序列化数据)和 HoloLens 应用程序(使用 Unity 和 VS 开发)共享,后者接收该数据并尝试使用相同的 dll 对其进行反序列化。

问题在于,当 WPF 应用程序序列化结构时,Marshal.SizeOf(str) 返回的大小为 12。而 Unity C# 脚本中完全相同的 dll 函数 app 返回的大小为 24。此外,数组中的最后 12 个字节都是 0。

因此,尝试反序列化来自 WPF 应用程序的对象时会出现超出范围的异常,因为它们的大小是 Unity 脚本中预期大小的一半。

这是我正在序列化的结构示例:

[Serializable]
public struct SimulationVariableModel

    public string Name  get; set; 
    public string Category  get; set; 
    public string ObjectId  get; set; 

这是将结构序列化为字节数组的函数:

    public static byte[] StrucToByteArray<T>(T str) where T : struct
    
        // In Unity: size = 24
        // In standalone WPF and UWP application: size = 12
        int size = Marshal.SizeOf(str);
        byte[] arr = new byte[size];

        IntPtr ptr = Marshal.AllocHGlobal(size);

        try
        
            Marshal.StructureToPtr(str, ptr, false);
            Marshal.Copy(ptr, arr, 0, size);
        
        finally
        
            Marshal.FreeHGlobal(ptr);
        

        return arr;
    

以及将字节数组转换为原始结构的函数:

    public static T ByteArrayToStruct<T>(byte[] arr) where T : struct
    
        T str = default;

        int size = Marshal.SizeOf(str);
        IntPtr ptr = Marshal.AllocHGlobal(size);

        try
        
            Marshal.Copy(arr, 0, ptr, size);
            str = (T)Marshal.PtrToStructure(ptr, str.GetType());

         finally
        
            Marshal.FreeHGlobal(ptr);
        

        return str;
    

这是在 Unity 中接收序列化数据并尝试将其反序列化为原始结构的事件处理程序:

private void Client_MqttMsgPublishReceived(object sender, MqttMsgPublishEventArgs e)

    if (e.Topic == NewSimulationVariableTopic) 

        // Dummy item to test that the byte Array length is not due to extra data in e.Message.
        SimulationVariableModel test = new SimulationVariableModel  Name = "Bla", Category = "Bla" ;

        // testByteArray.Length = 24
        byte[] testByteArray = StrucToByteArray(test);

        // De-serialization is successful
        test = ByteArrayToStruct<SimulationVariableModel>(testByteArray);

        // e.Message is the result from serialization 
        // using StructToByteArray<SimulationVariableModel>(str) in standalone WPF and UWP app.
        // De -serialization fails because e.Message.Length = 12
        SimulationVariableModel simVar = ByteArrayToStruct<SimulationVariableModel>(e.Message);

        // Logic with SimulationVariableModel
    

我认为这可能是由 Unity 环境引起的?我测试了一个 UWP 独立应用程序,它将结构序列化为 12 个字节而不是 24 个字节。

是否有人知道可能发生的情况以及如何解决此问题?

非常感谢。

【问题讨论】:

字符串通过像这样的“简单”路由序列化是出了名的尴尬,因为它们不是就地数据;坦率地说,我想知道你是否会更好地使用定义明确的序列化程序 - JSON、protobuf 等 我猜这与 string 数据类型在一个中被视为 Unicode(每个字符 2 个字节)而不是另一个(即 ASCII,每个字符 1 个字节)有关? 我建议使用接口(而不是 where T : struct 使用 where T : IMySerializable )这样您就可以为不同的数据类型编写自己的逻辑 您是直接从 CLR 使用的内存中复制字节来表示结构,这不是“真正的”序列化,它可能仅在“两端”在类似环境中运行时才有效(即使用相同的 CLR)。如果您只是想序列化数据,那么所有编组和指针处理对我来说似乎有点奇怪......您是否考虑过使用“传统”序列化? (例如使用XmlSerializerBinaryFormatter 或某种 JSON 库) c#中的一个字符是两个字节。使用 WCHAR(两个字节)或 CHAR(一个字节)的 WPF 也是如此。我怀疑你使用的是 T "char" 而不是 "byte" 【参考方案1】:

正如一些 cmets 所建议的,字符串在两个平台中的处理方式不同。我通过指定结构在其定义中的序列化方式解决了这个特定问题。示例:

[Serializable]
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi, Pack = 1)]
public struct SimulationVariableModel

    [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
    public readonly string Name;

    [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
    public readonly string Category;

    [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
    public readonly string Id;

    public SimulationVariableModel(string name, string category, string objectId)
    
        Name = name;
        Category = category;
        Id = objectId;
    

但是,对于更强大的解决方案,我最好使用 Mark Gravell 和 bassfader 建议的实际序列化器。

干杯!

【讨论】:

以上是关于不同应用程序(WPF 和 Unity)中的 C# struct 序列化返回不同大小的字节数组的主要内容,如果未能解决你的问题,请参考以下文章

C# WinForm程序中使用Unity3D控件 (转)

在 WPF 应用程序中嵌入 Unity3D 应用程序

与 C# 控制台应用程序中的相同代码相比,Regex.Match 在 Unity 中返回不同/错误的结果

将 Unity 中的 XAML (WPF) 应用与 MixedRealityToolkit 相结合

C# WPF 想要将数据存储在一个类中,并在不同 wpf 窗口中的多个不同类中使用

C# WPF MVVM模式Prism框架从零搭建(经典)