自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

EntityFrameworkCore上下文如何實(shí)現(xiàn)繼承?

開(kāi)發(fā) 前端
如果我們存在基礎(chǔ)設(shè)施服務(wù)和其他服務(wù),我們會(huì)定義屬于基礎(chǔ)設(shè)施服務(wù)的上下文以及其他服務(wù)的上下文, 而且會(huì)獨(dú)立部署,此時(shí)其他服務(wù)需要使用基礎(chǔ)服務(wù),我們都會(huì)暴露基礎(chǔ)服務(wù)接口給到其他服務(wù)調(diào)用,這也是常規(guī)操作。

?若在項(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ì)!?

責(zé)任編輯:武曉燕 來(lái)源: JeffckyShare
相關(guān)推薦

2022-09-14 13:13:51

JavaScript上下文

2017-05-11 14:00:02

Flask請(qǐng)求上下文應(yīng)用上下文

2024-01-29 08:49:36

RAG模型檢索

2012-12-31 10:01:34

SELinuxSELinux安全

2024-03-14 08:11:45

模型RoPELlama

2017-06-27 18:52:05

TensorFlow深度學(xué)習(xí)

2012-08-10 13:32:08

.NETAOP架構(gòu)

2023-07-11 10:02:23

2024-09-30 14:10:00

2017-12-17 17:01:23

限界上下文系統(tǒng)模型

2022-10-28 16:24:33

Context上下文鴻蒙

2025-03-18 08:14:05

2020-07-24 10:00:00

JavaScript執(zhí)行上下文前端

2021-07-26 07:47:36

Cpu上下文進(jìn)程

2010-02-25 17:04:54

WCF實(shí)例上下文

2019-05-06 14:36:48

CPULinux寄存器

2022-04-24 15:37:26

LinuxCPU

2025-04-07 01:02:00

GoAPI語(yǔ)言

2012-07-30 16:29:40

架構(gòu)架構(gòu)模式.NET

2011-06-28 10:55:02

QT QMainWindo 內(nèi)存泄露
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)