npgsql 处理json多态问题

目前.net版本为 9.0, 虽然 EFCore 7.0已经支持json, 但是对多态并不友好,所以目前的json映射依然使用 npgsql提供的方式。

解决问题

  • EFCore 对json多态支持不好
  • npgsql 处理json多态时,默认json顺序找不到metadata
  • 这个metadata指的是C#序列化多态类型时,默认对多态的数据插入$type属性来标记特定的子类型

模型配置

C# 复制代码
public class Marker
{
    public Guid Id { get; set;}
    
    [Column(TypeName = "jsonb")]
    public Meta[] Metas { get; set;}
}

[JsonPolymorphic(TypeDiscriminatorPropertyName = "type")] // 元数据的名称为 type
[JsonDerivedType(typeof(MetaImageList), "image-list")]    // 子类型元数据值 "type"="image-list"
[JsonDerivedType(typeof(MetaUrl), "url")]
public class Meta 
{
    public string Name { get; set; }
}

public class MetaUrl : Meta
{
    public string Data { get; set; }
}

public class MetaImageList : Meta
{
    public string[] Data { get; set; }
}

配置DbContext

arduino 复制代码
public class AppDbContext(DbContextOptions<AppDbContext> options) : DbContext(options)
{
    public DbSet<Marker> Markers => Set<Marker>();
}

依赖注入配置EF Core

C# 复制代码
var services = new ServiceCollection().AddDbContext<AppDbContext>(builder =>
{
    builder.UseNpgsql("connection_string", o =>
    {
        o.ConfigureDataSource(ob =>
        {
            // 支持json
            ob.EnableDynamicJson();
            
            ob.ConfigureJsonOptions(
                new JsonSerializerOptions 
                { 
                    // 小驼峰
                    PropertyNamingPolicy = JsonNamingPolicy.CamelCase,   
                    
                    // 重要 重要 重要 :允许元数据属性出现在 JSON 对象中的任意位置
                    AllowOutOfOrderMetadataProperties = true
                });
        });
    });
});

使用

C# 复制代码
var sp = services.BuildServiceProvider();
var dbcontext = sp.GetRequiredService<AppDbContext>();
dbcontext.Database.EnsureCreated();

dbcontext.Markers.Add(new Marker
{
    Id = Guid.CreateVersion7(),
    Metas = [new MetaUrl {
        Name = "bilibili",
        Data = "https://www.bilibili.com/"
    },new MetaImageList{
        Name = "图片组",
        Data = ["http://www.example.com/1.jpg"]
    }]
});
dbcontext.SaveChanges();

var markers = dbcontext.Markers.ToList();
Console.WriteLine(JsonSerializer.Serialize(markers, 
    new JsonSerializerOptions { 
        PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
        Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping,
        WriteIndented = true }));

成果

相关推荐
一 乐4 小时前
婚纱摄影网站|基于ssm + vue婚纱摄影网站系统(源码+数据库+文档)
前端·javascript·数据库·vue.js·spring boot·后端
玩泥巴的5 小时前
飞书 .NET SDK 事件处理的幂等性与去重机制
c#·.net·二次开发·飞书
在路上看风景5 小时前
3.2 FileStream
c#
码事漫谈5 小时前
Protocol Buffers 编码原理深度解析
后端
码事漫谈5 小时前
gRPC源码剖析:高性能RPC的实现原理与工程实践
后端
zwm2698888155 小时前
6号楼 部分声光24v电压达不到,显示11v
c#
踏浪无痕7 小时前
AI 时代架构师如何有效成长?
人工智能·后端·架构
程序员小假7 小时前
我们来说一下无锁队列 Disruptor 的原理
java·后端
武子康8 小时前
大数据-209 深度理解逻辑回归(Logistic Regression)与梯度下降优化算法
大数据·后端·机器学习
maozexijr9 小时前
Rabbit MQ中@Exchange(durable = “true“) 和 @Queue(durable = “true“) 有什么区别
开发语言·后端·ruby