外键拆分手记

我习惯性使用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/p/17357900.html

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

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

相关文章

  • ASP.Net执行cmd命令的实现代码

    ASP.NET执行cmd命令的实现,需要使用System.Diagnostics.Process类,该类可以让你启动一个新的进程,并且可以通过StandardInput输入命令,通过StandardOutput输出执行结果。以下是实现步骤: 1. 引入命名空间 using System.Diagnostics; 2. 创建Process对象并设置属性 Pro…

    C# 2023年5月31日
    00
  • C# 重写Notification提示窗口的示例代码

    下面是详细讲解“C# 重写Notification提示窗口的示例代码”的完整攻略: 一、什么是Notification提示窗口 Notification提示窗口就是Windows操作系统的一个提示框,一般用于通知用户系统的一些状态变化或提示信息。它一般弹出在屏幕的右下角,显示一定的时间之后会自动消失。 二、怎样重写Notification提示窗口 要重写No…

    C# 2023年6月7日
    00
  • C#中实现Fluent Interface的三种方法

    C#中实现Fluent Interface的三种方法攻略: 什么是Fluent Interface? Fluent Interface是一种编写API的方式,通过链式调用的语法方式,在代码中呈现出一种自然语言句子的形式。这种形式使得代码更加易读,易用,更具可扩展性。 方法一:基于接口实现 第一种方法是基于接口实现。通过使用C#中的接口和扩展方法,我们可以使得…

    C# 2023年6月3日
    00
  • Unity ScrollView实现无限循环效果

    下面是详细的“Unity ScrollView实现无限循环效果”的攻略: 1. 原理介绍 ScrollView是Unity中非常常用的滚动显示组件,但是如果要实现无限循环效果,需要对ScrollView进行改造。基本的思路是:在原有的ScrollView上拓展一个滑动方向的循环缓存列表,同时动态修改ScrollView的内容来保证显示的物品时刻与缓存列表中的…

    C# 2023年6月3日
    00
  • unityZXing二维码的生成与扫描

    一、UnityZXing简介 UnityZXing是一个为Unity3D提供二维码扫描识别和生成的插件。它使用Google ZXing库实现了对二维码的快速识别和生成,可以在Unity项目中轻松地添加二维码功能。 二、UnityZXing的安装 打开Unity工程,在Assets菜单中选择“Import Package” -> “Custom Pack…

    C# 2023年6月3日
    00
  • C#设置与获取环境变量的方法详解

    C#设置与获取环境变量的方法详解 什么是环境变量 在计算机系统中,操作系统和应用程序都需要使用很多参数和选项,例如文件路径、可执行文件路径、系统配置参数等。管理这些参数和选项的重要方式之一就是通过环境变量。环境变量可以看作是全局的键值对,每个键值对都由一个名称和一个值组成,其中名称称为变量名,值则称为变量值。 设置与获取环境变量 设置环境变量 在C#中,可以…

    C# 2023年6月8日
    00
  • JWT.net 操作实践方法

    JWT.net 操作实践方法 JSON Web Token(JWT)是一种基于JSON格式的Web Token标准,用于在不信任的环境下对用户进行身份验证和授权。JWT.net是一个C#中的JWT实现库,本篇文章将介绍如何使用JWT.net进行JWT生成、验证、签名等操作。 安装 可以通过NuGet包管理器或者命令行安装JWT.net: Install-Pa…

    C# 2023年5月31日
    00
  • .NET Core 环境变量详解

    一、概述 软件从开发到正式上线,在这个过程中我们会分为多个阶段,通常会有开发、测试、以及上线等。每个阶段对应的环境参数配置我们会使用不同的参数。比如数据库的连接字符串,开发环境一般我们都是连接的测试库。以前这种情况通常是 COPY 两个同名的配置文件来进行处理,然后在本地就使用本地的配置,生产环境就使用生产环境的配置文件,十分麻烦。而 ASP .NET CO…

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