EntityFrameworkCore上下文如何實(shí)現(xiàn)繼承?
?若在項(xiàng)目較小的情況下且僅內(nèi)部調(diào)用等等,為免去重新定義基礎(chǔ)設(shè)施服務(wù)上下文以及模型等等,我們大可以將基礎(chǔ)設(shè)施服務(wù)上下文打成nuget包形式或項(xiàng)目引用方式等等,然后其他服務(wù)上下文繼承基礎(chǔ)設(shè)施上下文,如此這般,我們就可以操作基礎(chǔ)設(shè)施模型,那么我們應(yīng)該怎么做呢?
實(shí)現(xiàn)上下文繼承
我們從頭開(kāi)講,比如我們定義其他服務(wù)上下文以及模型等等
public class TestDbContext : DbContext
{
public TestDbContext(DbContextOptions<TestDbContext> options) : base(options)
{
}
public DbSet<Test> Tests { get; set; }
}
[Table("tests")]
public class Test
{
[Column("id")]
public int Id { get; set; }
[Column("name")]
public string Name { get; set; }
}
接下來(lái)我們使用控制臺(tái)程序注入上下文并查詢(xún)表數(shù)據(jù),最基本操作,無(wú)需我多言
static void Main(string[] args)
{
var services = new ServiceCollection();
services.AddDbContext<TestDbContext>(options =>
{
options.UseSqlServer("Data Source=.;Initial Catalog=EFCore;User ID=sa;Password=sa123;");
});
var serviceProvider = services.BuildServiceProvider();
var context = serviceProvider.GetRequiredService<TestDbContext>();
var result = JsonConvert.SerializeObject(context.Tests.ToList());
}
此時(shí)上述服務(wù)上下文需要調(diào)用基礎(chǔ)服務(wù)上下文,我們?cè)撛趺崔k呢?先定義好基礎(chǔ)服務(wù)上下文
public class BaseDbContext : DbContext
{
public BaseDbContext(DbContextOptions<BaseDbContext> options) : base(options)
{
}
public DbSet<User> Users { get; set; }
}
[Table("users")]
public class User
{
[Column("id")]
public int Id { get; set; }
[Column("name")]
public string Name { get; set; }
[Column("birthdate")]
public DateTime BirthDate { get; set; }
[Column("address")]
public string Address { get; set; }
}
接下來(lái)我們將其他服務(wù)上下文TestDbContext繼承自上述基礎(chǔ)服務(wù)上下文
public class TestDbContext : BaseDbContext
{
public TestDbContext(DbContextOptions<TestDbContext> options) : base(options)
{
}
......
}
此時(shí)編譯會(huì)報(bào)CS1503錯(cuò)誤,無(wú)法將TestDbContext轉(zhuǎn)換為BaseContext,因?yàn)闃?gòu)造函數(shù)參數(shù)不匹配,我們知道DbContextOptions是DbContextOptions<T>父類(lèi),所以我們只需在BaseDbContext新增一個(gè)構(gòu)造函數(shù)即可
public class BaseDbContext : DbContext
{
public BaseDbContext(DbContextOptions<BaseDbContext> options) : base(options)
{
}
public BaseDbContext(DbContextOptions options) : base(options)
{
}
......
}
這樣一來(lái),我們則可以操作基礎(chǔ)服務(wù)上下文中的模型,如下
var context = serviceProvider.GetRequiredService<TestDbContext>();
var result = JsonConvert.SerializeObject(context.Users.ToList());
我們到這里是不是就大功告成了呢?當(dāng)然沒(méi)有,若此時(shí)通過(guò)基礎(chǔ)服務(wù)上下文直接操作,我們發(fā)現(xiàn)會(huì)拋出如下異常
啥意思呢?根據(jù)大致意思來(lái)看,就是說(shuō)上下文構(gòu)造函數(shù)有問(wèn)題,所以無(wú)法激活創(chuàng)建上下文,那么根本原因在哪里呢?這個(gè)問(wèn)題其實(shí)在此前博文有講解 ,甩出源碼如下:
private static Func<TContext> CreateActivator(DbContextOptions options)
{
var constructors
= typeof(TContext).GetTypeInfo().DeclaredConstructors
.Where(c => !c.IsStatic && c.IsPublic)
.ToArray();
if (constructors.Length == 1)
{
var parameters = constructors[0].GetParameters();
if (parameters.Length == 1
&& (parameters[0].ParameterType == typeof(DbContextOptions)
|| parameters[0].ParameterType == typeof(DbContextOptions<TContext>)))
{
return
Expression.Lambda<Func<TContext>>(
Expression.New(constructors[0], Expression.Constant(options)))
.Compile();
}
}
return null;
}
首先獲取上下文中聲明的構(gòu)造函數(shù)過(guò)濾掉了靜態(tài)和公共,且上下文必須有且只能有一個(gè)顯式構(gòu)造函數(shù)且參數(shù)只能為DbContextOptions<T>,我們恍然大悟,將新增的構(gòu)造函數(shù)訪問(wèn)修飾符修改為受保護(hù)的(protected)即可
public class BaseDbContext : DbContext
{
public BaseDbContext(DbContextOptions<BaseDbContext> options) : base(options)
{
}
protected BaseDbContext(DbContextOptions options) : base(options)
{
}
......
}
哦,沒(méi)啥可總結(jié)的勒,這玩意只能根據(jù)經(jīng)驗(yàn)猜或者看源碼可得知,再會(huì)!?