首页 > 其他分享 >问题记录:EntityFramework 一对一关系映射

问题记录:EntityFramework 一对一关系映射

时间:2024-09-30 10:52:02浏览次数:7  
标签:Extent1 映射 一对一 Teacher EntityFramework public Student Id Name

问题记录:EntityFramework 一对一关系映射

 

EntityFramework 一对一关系映射有很多种,比如主键作为关联,配置比较简单,示例代码:

public class Teacher
{
    public int Id { get; set; }
    public string Name { get; set; }
    public virtual Student Student { get; set; }
}

public class Student
{
    public int Id { get; set; }
    public string Name { get; set; }
    public virtual Teacher Teacher { get; set; }
}

上面代码表示 Teacher 和 Student 一对一关系,Fluent API 配置如下:

modelBuilder.Entity<Teacher>()
	.HasRequired(x => x.Student)
    .WithOptional(x => x.Teacher);
modelBuilder.Entity<Student>();

测试代码:

var teachers = await _teacherRepository.GetAll().Include(x => x.Student).ToListAsync()

生成 SQL 代码:

SELECT 
    [Extent1].[Id] AS [Id], 
    [Extent1].[Name] AS [Name], 
    [Extent2].[Id] AS [Id1], 
    [Extent2].[Name] AS [Name1]
    FROM  [dbo].[Teachers] AS [Extent1]
    INNER JOIN [dbo].[Students] AS [Extent2] ON [Extent1].[Id] = [Extent2].[Id]

另一种 Fluent API 配置如下:

modelBuilder.Entity<Teacher>();
modelBuilder.Entity<Student>()
	.HasRequired(x => x.Teacher)
    .WithOptional(x => x.Student);

执行同样测试代码,生成 SQL 代码:

SELECT 
    [Extent1].[Id] AS [Id], 
    [Extent1].[Name] AS [Name], 
    [Extent2].[Id] AS [Id1], 
    [Extent2].[Name] AS [Name1]
    FROM  [dbo].[Teachers] AS [Extent1]
    LEFT INNER JOIN [dbo].[Students] AS [Extent2] ON [Extent1].[Id] = [Extent2].[Id]

根据上面的测试情况,我们可以得到一些信息,首先测试代码查询 Teacher,然后 Inclue Student,Fluent API 配置的不同,生成的 SQL 代码也不同:

  • Fluent API 配置 Teacher,HasRequired Student 对应 INNER JOIN
  • Fluent API 配置 Student,HasRequired Teacher 对应 LEFT INNER JOIN

我们可以得出,一对一关系,Fluent API 只需要配置一个实体就可以了,根据查询关联的不同,配置对应的 HasRequired 和 WithOptional。

一对一关系,除了两个实体主键映射外,还有一种情况就是主键和外键映射,可以理解为主表和子表映射,示例代码:

public class Teacher
{
    public int Id { get; set; }
    public string Name { get; set; }
}

public class Student
{
    public int Id { get; set; }
    public string Name { get; set; }
    public int TeacherId { get; set; }
    public virtual Teacher Teacher { get; set; }
}

Student 中有一个 TeacherId 属性,对应 Teacher 中的主键 Id,在微软的官方示例中,Student 是作为主表,Teacher 作为子表,也就是说,我们在查询的时候是查询的 Student,然后 Include Teacher,Fluent API 配置:

modelBuilder.Entity<Teacher>();
modelBuilder.Entity<Student>()
	.HasRequired(x => x.Teacher)
    .WithMany()
    .HasForeignKey(x => x.TeacherId);

测试代码:

var students = await _studentRepository.GetAll().Include(x => x.Teacher).ToListAsync();

生成 SQL 代码:

SELECT 
    [Extent1].[Id] AS [Id], 
    [Extent1].[Name] AS [Name], 
    [Extent1].[TeacherId] AS [TeacherId], 
    [Extent2].[Id] AS [Id1], 
    [Extent2].[Name] AS [Name1]
    FROM  [dbo].[Students] AS [Extent1]
    INNER JOIN [dbo].[Teachers] AS [Extent2] ON [Extent1].[TeacherId] = [Extent2].[Id]

这是没有什么问题的,需要注意的是 Teacher 中并没有 Student 的导航属性,如果直接添加的话,运行会直接报错(Teachers 表默认生成的 Student_Id 字段找不到),解决方式是需要配置 Teacher 的相关 Fluent API。

上面的一对一关系,其实就是主表的一个子表扩展,在主表中存储子表的主键作为外键,查询的时候直接 Include 子表就可以了,但还有一种情况是,我查询子表,然后 Include 主表,主表的主键存储在子表中作为外键,这里的主表和子表概念只是相对的。

比如上面场景中,我查询 Teacher 然后把 Student Include 包含进来,如果是上面的配置是没有办法的,因为 Teacher 并没有配置导航属性,所以,我们需要改一下代码:

public class Teacher
{
    public int Id { get; set; }
    public string Name { get; set; }
    public virtual Student Student { get; set; }
}

public class Student
{
    public int Id { get; set; }
    public string Name { get; set; }
    public int TeacherId { get; set; }
    public virtual Teacher Teacher { get; set; }
}

上面说过,Teacher 增加 Student 导航属性会直接报错,然后我们再修改下 Fluent API 配置:

modelBuilder.Entity<Teacher>()
    .HasRequired(x => x.Student)
    .WithOptional(x => x.Teacher)
    .Map(x => x.MapKey("TeacherId"));
modelBuilder.Entity<Student>();

测试代码:

var teachers = await _teacherRepository.GetAll().Include(x => x.Student).ToListAsync();

生成 SQL 代码:

SELECT 
    [Extent1].[Id] AS [Id], 
    [Extent1].[Name] AS [Name], 
    [Extent1].[StudentCount] AS [StudentCount], 
    [Extent2].[Id] AS [Id1], 
    [Extent2].[Name] AS [Name1], 
    [Extent2].[TeacherId] AS [TeacherId]
    FROM  [dbo].[Teachers] AS [Extent1]
    INNER JOIN [dbo].[Students] AS [Extent2] ON [Extent1].[TeacherId] = [Extent2].[Id]

上面这段代码会执行报错的,因为[Extent1].[TeacherId] = [Extent2].[Id]的 Id 顺序错了,MapKey 配置的是 Teacher,而不是 Student,所以,我们再修改下 Fluent API 配置:

modelBuilder.Entity<Teacher>();
modelBuilder.Entity<Student>()
    .HasRequired(x => x.Teacher)
    .WithOptional(x => x.Student)
    .Map(x => x.MapKey("TeacherId"));

需要注意的是,因为 Teacher 中有了 Student 导航属性,所以我们没有办法再进行 HasForeignKey 的配置。

再次执行测试代码,并没有生成 SQL 代码,而是直接报错:Each property name in a type must be unique. Property name 'TeacherId' is already defined.

根据错误提示,我们去除 Student 中的 TeacherId 属性,重新执行测试代码。

生成的 SQL 代码:

SELECT 
    [Extent1].[Id] AS [Id], 
    [Extent1].[Name] AS [Name], 
    [Extent1].[StudentCount] AS [StudentCount], 
    [Extent3].[Id] AS [Id1], 
    [Extent3].[Name] AS [Name1], 
    [Extent3].[TeacherId] AS [TeacherId]
    FROM   [dbo].[Teachers] AS [Extent1]
    LEFT OUTER JOIN [dbo].[Students] AS [Extent2] ON [Extent1].[Id] = [Extent2].[TeacherId]
    LEFT OUTER JOIN [dbo].[Students] AS [Extent3] ON [Extent1].[Id] = [Extent3].[TeacherId]

结果是没有什么问题的,但 LEFT OUTER JOIN 了两次,不知道具体是什么原因。

网上找了相关的资料,但一对一关系示例都是那种:子表没有导航属性,主表存储子表的主键作为外键,并有子表的导航属性,上面的类似示例,stackoverflow 找到一个,但评论中的解决方式试过不行。

针对这种情况,如果有更好的实现方式,欢迎告知。

标签:Extent1,映射,一对一,Teacher,EntityFramework,public,Student,Id,Name
From: https://www.cnblogs.com/sexintercourse/p/18441434

相关文章

  • 爱与恨的抉择:ASP.NET 5+EntityFramework 7
    爱与恨的抉择:ASP.NET5+EntityFramework7  EF7的纠缠ASP.NET5的无助忘不了你的好一开始列出的这个博文大纲,让我想到了很久之前的一篇博文:恋爱虽易,相处不易:当EntityFramework爱上AutoMapper,只不过这次的剧情换主角了,而且与EF和AutoMapper爱情故事不同的是,这次是......
  • EntityFramework.Extended 支持 MySql
    EntityFramework.Extended支持MySql EntityFramework.Extended默认不支持MySql,需要配置如下代码:[DbConfigurationType(typeof(DbContextConfiguration))]//增加配置publicclassSchoolDbContext:DbContext,IDbContext{publicSchoolDbContext()......
  • 虚拟机端口映射到本机端口(超详细)
    应用场景:在工作或者学习中,我们和我们的团队都是处在一个局域网中的,为了某些业务,我们会在自己的本机安装虚拟机,比如我们会安装一台linux虚拟机作为服务器,在服务器上搭建对应的服务,这时候需要团队都能访问到你的虚拟机,但很明显,如果我们用的NAT模式的话,我们的虚拟......
  • 南沙csp-j/s一对一家教 解一本通题: 1937:【06NOIP普及组】数列
    ​【题目描述】给定一个正整数k(3≤k≤15),把所有k的方幂及所有有限个互不相等的k的方幂之和构成一个递增的序列,例如,当k=3时,这个序列是:1,3,4,9,10,12,13,…请你求出这个序列的第N项的值(用10进制数表示)。例如,对于k=3,N=100,正确答案应该是981。【输入】只有1行,为2个正整数,用一个......
  • 服务器数据恢复—存储中raid硬盘故障导致映射到服务器上的卷挂载不上的数据恢复案例
    服务器存储数据恢复环境&故障:一台存储上有一组由16块FC硬盘组建了一组raid。存储前面板上的对应10号和13号硬盘的故障灯亮起,存储映射到redhatlinux操作系统服务器上的卷挂载不上,业务中断。服务器存储数据恢复过程:1、通过存储的管理后台查看当前存储状态,管理后台报告逻辑卷状态......
  • 信息学奥赛复赛复习04-CSP-J2019-04-加工零件-位运算、整数映射0或1、结构体、初始化
    PDF文档回复:20240926<12019CSP-J题目4加工零件[题目描述]凯凯的工厂正在有条不紊地生产一种神奇的零件,神奇的零件的生产过程自然也很神奇。工厂里有n位工人,工人们从1∼n编号。某些工人之间存在双向的零件传送带。保证每两名工人之间最多只存在一条传送带如果......
  • 信息学奥赛复赛复习04-CSP-J2019-04-加工零件-位运算、整数映射0或1、结构体、初始化
    PDF文档公众号回复关键字:2024092612019CSP-J题目4加工零件[题目描述]凯凯的工厂正在有条不紊地生产一种神奇的零件,神奇的零件的生产过程自然也很神奇。工厂里有n位工人,工人们从1∼n编号。某些工人之间存在双向的零件传送带。保证每两名工人之间最多只存在一条传送带......
  • Leetcode 706. 设计哈希映射
    1.题目基本信息1.1.题目描述不使用任何内建的哈希表库设计一个哈希映射(HashMap)。实现MyHashMap类:MyHashMap()用空映射初始化对象voidput(intkey,intvalue)向HashMap插入一个键值对(key,value)。如果key已经存在于映射中,则更新其对应的值value。intget(in......
  • [Python手撕]实现哈希映射
    classNode:def__init__(self,key,value,next=None):self.key=keyself.value=valueself.next=nextclassMyHashMap:def__init__(self):self.array=[None]*(10**3)defput(self,key:int,value:int)->......
  • C# .net 8 used Pomelo.EntityFrameworkCore.MySql
    1.dotnetaddpackagePomelo.EntityFrameworkCore.MySqlusingMicrosoft.EntityFrameworkCore;namespaceConsoleApp84{internalclassProgram{staticvoidMain(string[]args){using(varcontext=newDbBookDataContex......