如何使用EF优雅的配置一对一的关系
在这两天的时间已经有两位同事问到EF(Code First)如何配置一对一的关系,这个说难也不难,说简单吧,一旦设计跑偏那么在Coding的过程中将会很痛苦。
先举个很简单的例子,两个类User和Profile,User里面存在用户的基本信息比如邮箱和密码,Profile里面存放用户的个人资料。
public class User
{
public int Id { get; set; }
public string Email { get; set; }
public string Password { get; set; }
public virtual Profile Profile { get; set; }
}
public class Profile
{
public int Id { get; set; }
public string Name { get; set; }
public Gender Gender { get; set; }
public string Phone { get; set; }
public string Address { get; set; }
public virtual User User { get; set; }
}
上面的代码应该是直接了当的。透过领域模型应该能充分描述出User和Profile的业务。分析:对于一个User来说只能有一个Profile(一对一),反过来对于一个Profile来说只能对应一个User(一对一),而且也必须有一个User。下面是通过fluent api做一些映射
public class UserMapping : EntityTypeConfiguration<User>
{
public UserMapping()
{
ToTable("tb_User");
}
}
public class ProfileMapping : EntityTypeConfiguration<Profile>
{
public ProfileMapping()
{
ToTable("tb_Profile");
HasKey(u => u.Id)
.HasRequired(u => u.User) // 对于一个Profile来说必须有一个User否则Profile无家可归
.WithOptional(u => u.Profile) // 同事一个User对应一个Profile 非必须
.WillCascadeOnDelete(false); // 不需要级联删除
}
}
上面的代码也应该也是直接了当的。那么这样生成的数据库结构是什么样的呢?约束又是什么样的呢?又是否符合我们的业务需求

User表没有什么问题,Profile的ID字段不单单是一个主键,同时也是一个外键。我们回到具体的业务中来。当我插入一条User数据的时候要不要插入Profile数据呢?其实是都可以的,如果同时插入的话,User和Profile应该是一起往前走的,而且User的ID和Profile的ID是一样的。那如果插入User的时候不插入Profile呢,其实也没有问题,详细的说明:
假如我插入两条User数据,User表会有两条数据,ID分别为1和2。Profile表为空,没有数据。2用户有一天回过头来想完善自己的个人详细信息,插进去的Profile数据ID应该为2,后来1用户也来完善自己的个人资料,这个时候插入的Profile数据ID为1(虽然2用户先完善的,2用户的ProfileID还是2,1用户的ProfileID还是1)。
static void Main(string[] args)
{
DemoDbContext db = new DemoDbContext();
var user = db.Users.SingleOrDefault(u => u.Id == 2);
Profile profile = new Profile
{
Address = "beijing"
};
user.Profile = profile;
db.SaveChanges();
}
顺便对领域模型唠叨几句:
领域模型是领域驱动设计中最重要的对象,它们是描述我们业务的对象,应该最大力度保持干净,整洁。作为一名开发人员应该多花点时间放在领域的设计上。然而领域的设计应该是和领域专家(产品经理)强度沟通的情况下去完成设计的。比如要做一款财务软件,程序员怎么可能对财务非常精通,都是和财务专家或懂财务的人沟通的基础上去完成我们的领域设计。所以在公司中应确保最好,最有经验的开发人员分配到领域相关的任务上去。
如何使用EF优雅的配置一对一的关系的更多相关文章
- EF Code-First 学习之旅 配置一对一的关系
1对1.1对0 的关系 例如:Entity1与零个或一个Entity2的实例有关系 public class Student { public Student() { } public int Stu ...
- 11.翻译系列:在EF 6中配置一对零或者一对一的关系【EF 6 Code-First系列】
原文链接:https://www.entityframeworktutorial.net/code-first/configure-one-to-one-relationship-in-code-fi ...
- 9.Configure One-to-One(配置一对一关系)【Code-First系列】
现在,开始学习怎么配置一对一的关系,众所周知,一对一的关系是:一个表中的主键,在另外一个表中,同时是主键和外键[实际上是一对零或者一对一]. 请注意:一对一的关系,在MS SQL Server中,技术 ...
- 【EF Code First】 一对一、一对多的多重关系配置
这里使用相册Album和图片Picture的关系做示例 1,Album与Picture最基本的关系是1-n(一个相册可以有多张图片) 这时Album.Picture实体类可以这么定义 /// < ...
- 13.翻译系列:Code-First方式配置多对多关系【EF 6 Code-First系列】
原文链接:https://www.entityframeworktutorial.net/code-first/configure-many-to-many-relationship-in-code- ...
- 8.翻译系列: EF 6中配置领域类(EF 6 Code-First 系列)
原文地址:http://www.entityframeworktutorial.net/code-first/configure-classes-in-code-first.aspx EF 6 Cod ...
- 12.翻译系列:EF 6 中配置一对多的关系【EF 6 Code-First系列】
原文链接:https://www.entityframeworktutorial.net/code-first/configure-one-to-many-relationship-in-code-f ...
- EF实体类配置总结
实体类配置总结 Entity Framework 6 Code First 实践系列(1):实体类配置总结 2014-03-25 12:58 by TJerry, 719 阅读, 6 评论, 收藏, ...
- Habernate配置一对一,一对多,多对多(二)
一.开篇 紧接着上篇的博客来写:http://www.cnblogs.com/WJ--NET/p/7845000.html(habernate环境的搭建) 二.配置一对一 2.1.新建客户类和公司类( ...
随机推荐
- Shopex如何清理缓存
一.进入后台,点击 右上角 的"关于" 二.点击:缓存系统: 三.点击"清空缓存" 四.清除成功!
- POJ 3259 Wormholes Bellman_ford负权回路
Description While exploring his many farms, Farmer John has discovered a number of amazing wormholes ...
- 到底什么样的企业才适合实施SAP系统?
SAP系统作为全宇宙第一的ERP,号称世界500强里面有80%的企业部署了SAP系统,总部位于德国沃尔多夫市,在全球拥有6万多名员工,遍布全球130个国家,并拥有覆盖全球11,500家企业的合作伙伴网 ...
- jquery左右轮播
<!--------html代码:-----------> <!DOCTYPE html><html><head><title>carous ...
- c++ static用法总结【转载】
static关键字是C, C++中都存在的关键字.static从字面理解,是“静态的“的 意思,与此相对应的,应该是“动态的“. static的作用主要有以下3个: 1.扩展生存期: 2.限制作用域: ...
- ACM退役帖 -- 未真正开始也不会结束
2017.5.21,20岁的最后一天,昨天,随着2017年安徽省大学生程序设计竞赛落下帷幕,我也正式退役了ACM了.连ACM区域赛也没去过的我,也许是不够格提出退役ACM这句话的,但对ACM的热爱,虽 ...
- Java中parse()和valueOf(),toString()的区别
1.parse()是SimpleDateFomat里面的方法,你说的应该是parseInt()或parsefloat()这种方法吧, 顾名思义 比如说parseInt()就是把String类型转化为i ...
- 统计s="hello alex alex hello haiyan cc haiyan com"中每个单词的个数
这个题可以有好几种解题方法. 一.索引值获取 s="hello alex alex hello haiyan cc haiyan com" l=s.split() dic={} f ...
- Kotlin 初窥门径[2]:流程控制
流程控制语句是编程语言中的核心之一.可以分为分支语句.循环语句和跳转语句.本文将详细介绍一下 Kotlin 中的流程控制语句. If 表达式 在Kotlin中一切都是表达式,也就是说一切都返回一个值. ...
- Windows MDI(Multiple-Document Interface)
Windows多文档窗口编程中,需要注意的以下几点: 1.主窗口与文档窗口之间还有一个Client Window. 2.创建文档窗口.通常认为创建子窗口就用CreateWindow,但是MDI中创建文 ...