.net 6 efcore一个model映射到多张表(非使用IEntityTypeConfiguration)

现在有两张表,结构一模一样,我又不想创建两个一模一样的model,就想一个model映射到两张表

废话不多说直接上代码

  1. 安装依赖包
  2. 创建model
csharp 复制代码
namespace oneModelMultiTable.Model
{
    public class Test
    {
        public int id { get; set; }

        public string name { get; set; }

        public string tablename { get; set; }
    }
}
  1. 创建DBContext
    我们需要使用tablename 动态指定表名,因此需要在DBContext中添加这个属性
csharp 复制代码
namespace oneModelMultiTable
{
    public class DBHelper:DbContext
    {
        public DbSet<Test> testConfigs { get; set; }

        public string tablename { get; set; }
        public DBHelper(DbContextOptions<DBHelper> options):base(options)
        {

        }
        protected override void OnModelCreating(ModelBuilder modelBuilder)
        {

            modelBuilder.Entity<Test>(b =>
            {
                b.ToTable(tablename);
                b.HasKey(p => p.id);
            });

            base.OnModelCreating(modelBuilder);

            //modelBuilder.ApplyConfigurationsFromAssembly(this.GetType().Assembly);
        }
    }
}
  1. 创建DynamicModelCacheKeyFactory 继承IModelCacheKeyFactory
csharp 复制代码
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
namespace oneModelMultiTable
{
    public class DynamicModelCacheKeyFactory : IModelCacheKeyFactory
    {

        public object Create(DbContext context, bool designTime)
        {
            object p = context is DBHelper dynamicContext
                ? (context.GetType(), dynamicContext.tablename)
                : (object)context.GetType();
            return p;
        }
    }
}
  1. 依赖注入
csharp 复制代码
builder.Services.AddDbContext<DBHelper>(options =>
{
    options.UseNpgsql(@"Host=192.168.214.133;Port=32222;Database=postgresdb;Username=postgresadmin;Password=admin123")
    .ReplaceService<IModelCacheKeyFactory, DynamicModelCacheKeyFactory>();
});
  1. 创建controller
csharp 复制代码
using Microsoft.AspNetCore.Mvc;
using oneModelMultiTable.Model;

// For more information on enabling Web API for empty projects, visit https://go.microsoft.com/fwlink/?LinkID=397860

namespace oneModelMultiTable.Controllers
{
    [Route("api/[controller]/[action]")]
    [ApiController]
    public class TestController : ControllerBase
    {
        public readonly DBHelper dBHelper;

        public TestController(DBHelper _DBHelper)
        {
            dBHelper = _DBHelper;
        }

        // GET: api/<TestController>
        [HttpGet]
        public List<Test> Get(string tablename)
        {
            dBHelper.tablename = tablename;
            return dBHelper.testConfigs.ToList();
        }
    }
}

原理

你可能想通过ToTable()方法来更改表名,但是我们如何在OnModelCreating方法中更改表名呢?当EF构建模型时,它只运行OnModelCreating一次。

对于这个场景,我们需要通过使用IModelCacheKeyFactory 来更改默认映射,它允许我们连接到模型缓存机制,以便EF可以根据其属性创建不同的模型。

EF使用IModelCacheKeyFactory为模型生成缓存键。

https://yanxiaodi.medium.com/mapping-the-model-to-multiple-tables-with-entityframework-core-b46bdeed8661

https://medium.com/@pawel.gerr/entity-framework-core-changing-database-schema-at-runtime-dcf1211768c6

https://github.com/xdqt/asp.net-core/tree/master/oneModelMultiTable

相关推荐
公子小六7 分钟前
推荐一种手动设置异步线程等待机制的解决方案
windows·microsoft·c#·.net
zzlyx993 小时前
.NET采用Avalonia11+SukiUI6.0.2开发桌面程序常用对话框
.net·avalonia·sukiui
一个帅气昵称啊4 小时前
在.NET中如何优雅的使用DotNetCore.CAP实现分布式事务,事件总线和消息最终一致性
分布式·微服务·.net
三天不学习4 小时前
Xamarin 与 .NET MAUI:.NET跨平台原生移动App开发前世今生
.net·xamarin·.net maui
a***81394 小时前
深入浅出 SQLSugar:快速掌握高效 .NET ORM 框架
.net
Crazy Struggle5 小时前
基于 JSON 配置的 .NET 桌面应用自动更新方案
.net·winform·自动更新
c***87195 小时前
【update 更新数据语法合集】.NET开源ORM框架 SqlSugar 系列
开源·.net
唐青枫6 小时前
C# 原始字符串字面量全面解析:多行字符串终于优雅了!
c#·.net
缺点内向9 小时前
如何在 C# 中将 Excel 工作表拆分为多个窗格
开发语言·c#·.net·excel
夏霞16 小时前
c# 使用vs code 创建.net8.0以及.net6.0 webApi项目的教程
开发语言·c#·.net