前言
假设有一个 Person 抽象基类,其中包含 Student 和 Teacher 派生类:
public class Person
{public string Name { get; set; }
}public class Student : Person
{public int Score { get; set; }
}public class Teacher : Person
{public string Title { get; set; }
}
如果 API 返回类型是单个 Person 抽象基类,System.Text.Json
内置功能已经可以直接支持序列化派生类:
[HttpGet]
[Route("get")]
public Person Get()
{Person person = new Student { Name = "zhangsan", Score = 100 };return person;
}
但是,当 API 需要返回 Person 抽象基类的集合时,System.Text.Json
就不知道如何处理了,即使集合内部只有一种派生类型:
[HttpGet]
[Route("list")]
public IEnumerable<Person> List()
{return new Person[] {new Student { Name = "zhangsan", Score = 100 }};
}
可以看到,只处理了基类的属性。
我们必须主动告诉System.Text.Json
如何处理序列化派生类,因此需要自定义转换器。
解决方案
为基类创建自定义转换器:
public class PersonConverter : JsonConverter<Person>
{public override Person Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options){throw new NotImplementedException();}public override void Write(Utf8JsonWriter writer, Person person, JsonSerializerOptions options){writer.WriteStartObject();if (person is Student student){writer.WriteNumber("score", student.Score);}else if (person is Teacher teacher){ writer.WriteString("title", teacher.Title);}writer.WriteString("name", person.Name);writer.WriteEndObject();}
}
然后修改 Startup.cs 文件,注册自定义转换器:
services.AddControllers().AddJsonOptions(options =>options.JsonSerializerOptions.Converters.Add(new PersonConverter()));
运行,序列化成功:
上面的转换器代码通过手动写入每个属性实现序列化,那属性一多还不要累死!
一种替代方法是调用System.Text.Json
内置功能实现序列化单个派生类:
public override void Write(Utf8JsonWriter writer, Person person, JsonSerializerOptions options)
{JsonSerializer.Serialize(writer, (object)person, options);
}
注意,必须将要序列化的对象转换为 object
结论
通过自定义转换器,我们实现了使用System.Text.Json
序列化派生类。
添加微信号【MyIO666】,邀你加入技术交流群