外键拆分手记

我习惯性使用OData,它的$expand与层级查询非常好用,这个功能非常依赖于数据库的导航属性,也就是外键结构。最近想着把一个单体的系统拆分为多个小系统,首先需要处理外键依赖的问题。

多个服务各自有各自的数据库,数据库层面并不互通,也就无法使用外键约束。

我使用EF Core来描述数据库的结构,有两个实体类如下:


public class AD_Insect_Info
{
	[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]
	public int Id {get;set;}
	public string Name { get; set; }
	public virtual List<AD_Insect_Datum> Results { get; set; }
}

public class AD_Insect_Datum
{
	[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]
	public int Id { get; set; }
	public virtual AD_Insect_Info AttachDevice { get; set; }
	[NotMapped]
	public override string AttachId => AttachDevice.AttachDeviceId;
}

可以看到他们之间有一个导航属性,实现一个一对多的关系。如果系统是新系统,大可直接进行拆解,想怎么弄就怎么弄。但是对于已经有较多数据的现有系统,最好使用渐进拆分的方式。

思路:通过多次,每次只修改一点,维持对旧有系统的兼容性,并在完全拆分前保持系统高度可用,破坏性最小。

指定外键

跨数据库的访问,数据完整性不能由数据库通过外键实现,我们需要在应用层实现自己的逻辑。由于外键不存在,我们需要在数据表中有表示对外引用的字段。而对于导航属性而言,外键已经由EF Core自动建立,为了防止数据结构变化导致的问题,我们可以明确指定外键。

查询数据表结构

EF Core有默认的外键命名规则,通常是字段名+外键的字段名,对于本例,则是AttachDeviceId,保险起见,我们可以查询数据库获得外键列名称。

显式指定外键名称

导航属性的外键名字不是那么直观,我们可以使用EF Core的ForeignKey特性。

public class AD_Insect_Datum : AttachDataBase
{
	[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]
	public int Id { get; set; }
	[ForeignKey("AttachDeviceId")]
	public virtual AD_Insect_Info AttachDevice { get; set; }
	public int AttachDeviceId { get; set; }
}

这样,这个对象Id就被显式指定了。

查询替代

前一步指定了外键,但是实际上并没有对EF Core表做任何更改,原来的程序可以正常运行,我们需要在这个基础上继续改造,将使用导航属性的地方修改一下。

public virtual async Task<IActionResult> Data(string key)
{
	return Ok(_context.AD_Insect_Data.Where(w => w.AttachDevice.Name == key));
}

修改为:

public virtual async Task<IActionResult> Data(string key)
{
	var instances = _context.AD_Insect_Infos.Where(w=>w.Name == key).Select(w=> w.Id).ToList();
	return Ok(_context.AD_Insect_Data.Where(w => instances.Contains(w.AttachDeviceId)));
}

这样就避免使用导航属性依赖。

我这里也没有使用连表查询方法,因为将来需要拆分。

删除外键

由于不再使用导航属性,可以安全地删除外键。注意,可能需要补充一些业务逻辑以确保数据库中的数据完整性。

拆分DbContext

将来DbContext将不再拥有AD_Insect_Infos,因此我们需要进一步拆分。

public virtual async Task<IActionResult> Data(string key)
{
	IEnumerable<int> instances = Foreign.GetList(key);
	return Ok(_context.AD_Insect_Data.Where(w => instances.Contains(w.AttachDeviceId)));
}

public class Foreign
{
	public IEnumerable<int> GetList(string key)
	{
		return _context.AD_Insect_Infos.Where(w=>w.Name == key).Select(w=> w.Id).ToList();
	}
}

上面代码表示大致的思路,请自行处理注入依赖等问题。

然后就是熟悉的动作了,将Foreign类作为一个数据服务,只要返回的类型是IEnumerable<int>就可以了,数据的来源可以是本身,也可以是外部的服务。

原文链接:https://www.cnblogs.com/podolski/archive/2023/04/27/17357900.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:外键拆分手记 - Python技术站

(0)
上一篇 2023年4月25日
下一篇 2023年4月27日

相关文章

  • IE6下javasc#ipt:void(0) 无效的解决方法

    针对IE6下javasc#ipt:void(0)无效的解决方法,我们可以采取以下步骤: 1. 了解问题 在很多老版本的IE浏览器中,当我们使用 javascript:void(0) 作为超链接的href属性值时,某些情况下会出现链接无法点击的问题,比如在IE6下。因此需要寻找解决方法。 2. 解决方法 方法一:使用window.event.returnVal…

    C# 2023年6月7日
    00
  • C#实现简单串口通信

    为了实现串口通信,首先需要掌握C#中的串口通信相关类和方法。常用的类有SerialPort和SerialDataReceivedEventArgs,常用的方法有Open和Write等。下面分步骤讲解实现串口通信的完整攻略。 1. 新建C#控制台应用程序 首先,打开Visual Studio,新建一个C#控制台应用程序。选择“新建项目”–>“Visua…

    C# 2023年6月7日
    00
  • js实现C#的StringBuilder效果完整实例

    下面就是详细讲解“js实现C#的StringBuilder效果完整实例”的攻略: 1. 概述 String 类是 JavaScript 中非常重要的内置类,我们在编程中常常需要处理大量字符串的拼接,常见的做法是使用 + 运算符或者字符串模板等。但是这种方法在处理大量字符串时会极大降低性能,并且难以维护。 这时,我们可以使用类似于 C# 中的 StringBu…

    C# 2023年6月7日
    00
  • C#异常处理详解

    下面我将详细讲解“C#异常处理详解”的完整攻略。 什么是异常? 异常(Exception)是指在程序执行过程中出现的错误状况,如内存不足、数组越界、数学计算异常等。这些错误状况可能会导致程序异常终止,而异常处理就是解决这些问题的方法。 异常处理的方法 在C#中,我们可以使用try-catch语句来处理异常。try块中放置我们要执行的代码,如果在执行过程中出现…

    C# 2023年6月6日
    00
  • C#通过标签软件Bartender的ZPL命令打印条码

    下面我将详细讲解“C#通过标签软件Bartender的ZPL命令打印条码”的完整攻略。 1. Bartender软件的安装和使用 Bartender是一款功能强大的标签设计和打印软件,可以用于创建各种类型的标签和条码。首先需要在官网下载Bartender的安装包并完成安装。 使用Bartender进行标签设计和打印的具体流程如下: 打开Bartender软件…

    C# 2023年6月6日
    00
  • C#中委托用法实例详解

    下面是详细讲解“C#中委托用法实例详解”: 什么是委托 委托是一种类型,它可以用于封装对方法的引用。简单来说,委托就是存储了方法的对象,可以像对待方法一样对待它并向它传递参数。在C#中,使用委托可以大大简化回调函数,具有非常强的灵活性。 如何声明委托 使用delegate关键字可以声明一个委托,如下所示: delegate void MyDelegate(i…

    C# 2023年6月7日
    00
  • c# 识别图片格式的方法

    C#识别图片格式的方法 在C#中,我们可以使用System.Drawing.Image类来识别图片格式。 Image类使用一个ImageFormat枚举成员来保存图像的格式信息。我们可以通过检查这个成员,来确定图像的格式。 方法一:使用GetImageType方法 使用GetImageType方法可以方便地获取图像格式。以下是示例代码: using Syst…

    C# 2023年5月15日
    00
  • C#中的小数和百分数计算与byte数组操作

    C#中的小数和百分数计算与byte数组操作是常见的应用场景,下面我们将分别进行说明。 小数和百分数计算 小数计算 在C#中,小数计算使用double或decimal类型。这两种类型的区别在于表示的精度不同,double类型精度较低,decimal类型精度较高,如果需要保留小数位数比较多的情况下,建议使用decimal类型。 下面是一个示例,对两个小数进行加减…

    C# 2023年6月8日
    00
合作推广
合作推广
分享本页
返回顶部