c# - Newtonsoft.json 序列化和反序列化基本/继承,其中类来自共享项目

标签 c# json serialization json.net shared-project

所以我有两个类,如下所示。它们都位于同一命名空间和同一共享项目中。

public class Person{
   public string Name{get;set;}
}

public class EmployedPerson : Person{
   public string JobTitle{get;set;}
}

当我将这些项目序列化到rabbitmq中时,我将序列化为基类,如下所示:

JsonSerializerSettings settings = new JsonSerializerSettings
{
   TypeNameAssemblyFormatHandling = TypeNameAssemblyFormatHandling.Simple,
   TypeNameHandling = TypeNameHandling.Objects
};
JsonConvert.SerializeObject(input, settings)

但是,在反序列化时我遇到了问题。我希望能够执行如下所示的操作,将其反序列化为基类,然后检查它是否是继承类型。

类型检查:

Person person = Deserialize<Person>(e.Body, Encoding.Unicode);
   if (person is EmployedPerson)
   {
    logger.LogInformation("This person has a job!");
    }

反序列化设置:

   JsonSerializerSettings settings = new JsonSerializerSettings
   {
      TypeNameAssemblyFormatHandling = TypeNameAssemblyFormatHandling.Simple,
      TypeNameHandling = TypeNameHandling.Auto
   };

反序列化逻辑:

    private static T Deserialize<T>(byte[] data, Encoding encoding) where T : class
    {
        try
        {
            using (MemoryStream stream = new MemoryStream(data))
            using (StreamReader reader = new StreamReader(stream, encoding))
                return JsonSerializer.Create(settings).Deserialize(reader, typeof(T)) as T;
        }
        catch (Exception e)
        {
            Type typeParameter = typeof(T);
            logger.LogError(LogEvent.SERIALIZATION_ERROR, e, "Deserializing type {@TypeName} failed", typeParameter.Name);
            logger.LogInformation(Encoding.UTF8.GetString(data));
            return default(T);
        }
    }

结果: 上面的代码失败,因为 $type 属性包含程序集名称,并且在rabbitmq的每一端,程序集名称不同,因为这些类位于共享项目内。

错误示例:

Newtonsoft.Json.JsonSerializationException: Error resolving type specified in JSON 'Shared.Objects.EmployedPerson, Person.Dispatcher'. Path '$type', line 1, position 75. ---> System.IO.FileNotFoundException: Could not load file or assembly 'Person.Dispatcher, Culture=neutral, PublicKeyToken=null'. The system cannot find the file specified.

最佳答案

谢谢@dbc,据我所知,您编写自定义 SerializationBinder 的建议是解决我的问题的最佳解决方案。

我使用了 KnownTypesBinder,其实现位置为:https://www.newtonsoft.com/json/help/html/SerializeSerializationBinder.htm

已知类型绑定(bind)器:

public class KnownTypesBinder : ISerializationBinder
    {
        public IList<Type> KnownTypes { get; set; }

        public Type BindToType(string assemblyName, string typeName)
        {
            return KnownTypes.SingleOrDefault(t => t.Name == typeName);
        }

        public void BindToName(Type serializedType, out string assemblyName, out string typeName)
        {
            assemblyName = null;
            typeName = serializedType.Name;
        }
    }

将 SerializationBinder 设置为 KnownTypesBinder 实例的 JsonSerializerSettings 用于序列化和反序列化端点。我可能只需要它用于反序列化端,但为了一致性而将其放入两者中。

settings = new JsonSerializerSettings
{
    TypeNameAssemblyFormatHandling = TypeNameAssemblyFormatHandling.Simple,
    TypeNameHandling = TypeNameHandling.Objects,
    SerializationBinder = new KnownTypesBinder()
};

创 build 置对象后,我将其传递到 JsonConvert 序列化函数中。

JsonConvert.DeserializeObject<T>(Encoding.Unicode.GetString(input), settings) 

另请注意,KnownTypesBinder 中的 KnownType 必须预先填充您要反序列化的所有非基本类型。

编辑: 我目前不接受我自己的答案,因为我不知道如何处理复杂类型的列表。例如,如果一个 Person 有一个 List 和一个 List,当 typeName 为“List`1”时,您返回什么类型,也可以是其中之一。

编辑 以下版本的 KnownTypesBinder 解决了我与对象列表相关的问题。

public class KnownTypesBinder: ISerializationBinder
{
    public IList<Type> KnownTypes { get; set; }

    public Type BindToType(string assemblyName, string typeName)
    {
        return KnownTypes.SingleOrDefault(t => t.UnderlyingSystemType.ToString() == typeName);
    }

    public void BindToName(Type serializedType, out string assemblyName, out string typeName)
    {
        assemblyName = null;
        typeName = serializedType.UnderlyingSystemType.ToString();
    }
}

关于c# - Newtonsoft.json 序列化和反序列化基本/继承,其中类来自共享项目,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45923177/

相关文章:

c# - 在 PropertyGrid 上设置 SelectedTab

c# - Azure CDN 端点使用 https 提供 502 错误网关

javascript - 多选数据表中的行,如何从JSON获取一列数据到php

c# - 将 dataGridView 序列化为 JSON

c# - 使用 linq to entities 和自定义 IEqualityComparer 的不同计数

c# - 数组问题(无法将类型 'int' 隐式转换为 'int[]')

perl - 如何确定数组引用中的元素数量?

javascript - 下划线js查找嵌套json

java - 将 Java 对象保存到文件中

python - 如何在跨语言对象序列化中处理 ruby​​ 符号