这一篇说说 EF Fluent API 和 DataAnnotations

参考 :

http://msdn.microsoft.com/en-us/data/jj591617.aspx

http://msdn.microsoft.com/en-us/data/jj591620.aspx

Fluent API 和 DataAnnotations 主要作用是让我们定义数据库映射的关系还有一些比如名字不相等的情况等等。

关系数据库一般我们有 1对1,1对多,多对多的表关系

下面我们来一一列出

注 : 这里的思想是先有表,才有类,这里只是例子,真真开发的时候通常是建类图才建数据库的。

我给一个电子商务的案子

我们有个产品种类的表 (prod_category) 还有一个产品表 (prod) , 1个产品可以在多个种类中出现,1个种类也可以有多个产品,所以他们的关系是 (多对多)

所以呢,我们得要多一个表来维持它们的关系,就叫 prod_category_vs_prod 表吧.

prod_category表

[Table("prod_category")] //使用标签,我们可以修改表名
public class ProdCategory
{
[Key] //标签Key,代表id 是主键
public Int32 id { get; set; }
public string category { get; set; } //种类名字 public virtual ICollection<Prod> prods { get; set; } //一个种类有多个产品,所以这里是个 collection , 记得是virtual哦,原理以后会解释.
}

prod表

[Table("prod")]
public class Prod
{
[Key]
public Int32 id { get; set; }
public string code { get; set; }
public string name { get; set; }
public virtual ICollection<ProdCategory> categorys { get; set; } //这里也可以反向获取到种类集合,一个产品有很多种类
}

好,那我们来定义关系

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
modelBuilder.Entity<Prod>()
.HasMany(prod => prod.categorys) //prod 有很多 categorys
.WithMany(category => category.prods) //category 也有很多prods
.Map(m =>
{
m.ToTable("prod_category_vs_prod");
m.MapLeftKey("prod_category_id");
m.MapRightKey("prod_id");
});
}

.Map 记入了维护它们之间的关系表,也就是 prod_category_vs_prod 建对建

好,接着是 1对0,或1对1 关系

这里想不出特别好的例子,我们假设一个产品有一个或没有细节

细节被记入在一个 prod_detail 表中

[Table("prod_detail")]
public class ProdDetail
{
[Key]
public Int32 id { get; set; }
public string fullDescription { get; set; }
[Column("prod_id")] //用标签需改column名
public Int32 prodId { get; set; } //这个是外键 public virtual Prod prod { get; set; } //一个detail肯定有一个prod
}
    [Table("prod")]
public class Prod
{
[Key]
public Int32 id { get; set; }
public string code { get; set; }
public string name { get; set; } public virtual ProdDetail detail { get; set; } //这里有一个detail,但有可能是null
public virtual ICollection<ProdCategory> categorys { get; set; }
}
1-0
modelBuilder.Entity<ProdDetail>()
.HasKey(detail => detail.prodId); //一对一,所以prodId也算是主键
modelBuilder.Entity<ProdDetail>()
.HasRequired(detail => detail.prod) //detail 一定有 prod
.WithOptional(prod => prod.detail); //prod 不一定有 detail //1 -1
modelBuilder.Entity<ProdDetail>()
.HasKey(detail => detail.prodId); //也算主键
modelBuilder.Entity<Prod>()
.HasRequired(prod => prod.detail) //prod 一定有 detail
.WithRequiredPrincipal(detail => detail.prod); detail 也一定有 prod

最后是1对多关系

比如1个产品有很多颜色

就一个 prod_color表吧

    [Table("prod_color")]
public class ProdColor
{
[Key]
public Int32 id { get; set; }
public string color { get; set; }
[Column("prod_id")]
public Int32 prodId { get; set; } public virtual Prod prod { get; set; } //一定有prod
}

直接上关系代码

//1-n
modelBuilder.Entity<ProdColor>()
.HasRequired<Prod>(color => color.prod) //color 一定有 prod
.WithMany(prod => prod.colors) //prod 有很多 colors 
.HasForeignKey(color => color.prodId); //color 有一个外键 prodId

modelBuilder.Entity<Prod>()
.HasMany<ProdColor>(prod => prod.colors) //prod 有很多 colors 
.WithRequired(color => color.prod) //color 一定有 prod 
.HasForeignKey(color => color.prodId); //color 有外键 prodId

好啦,大致上是这样先。

其实 1-0|1, 1-n 都可以直接用 DataAnnotations 来完成,并不需要用 Fluent API .

只是多对多的关系好像不行,所以呢

其实2个都应该学会,我个人是觉得如果只用标签来描述全部东西也是有点乱,

所以呢,关系就用 Fluent API , 如果是表名字,类型,maxlength, primary key 等等可以用标签来完成。这样分担工作就不乱了咯!

这里给出常用的整理版本

1-1 关系

//DataAnnotation 1-1 table
//SQLtable : member , columns : memberId, name
[Table("member")]
public class Member
{
[Key]
public Int32 memberId { get; set; }
public string name { get; set; }
[Required] //1-1关系最好是放啦
public virtual Address address { get; set; }
}
//SQLtable : address , columns : memberId, postcode, country
[Table("address")]
public class Address
{
[Key, ForeignKey("member")] //关键就是这个关系连接, "member" 是 tableName
public Int32 memberId { get; set; } //这里写addressId 也行
public string postcode { get; set; }
public string country { get; set; }
public virtual Member member { get; set; }
}

Entity Framework with MySQL 学习笔记一(关系)的更多相关文章

  1. Entity Framework with MySQL 学习笔记一(关系整理版)

    1-1 设置 //DataAnnotation 1-1 | 1-0 table //SQLtable : member , columns : memberId, name //SQL basic l ...

  2. Entity Framework with MySQL 学习笔记一(安装)

    声明 :  数据库是Mysql,本人的程度只到会写sql语句(不会储蓄过程), c# 会基本的ADO.NET数据库访问,LINQ基础. 这篇只做个人学习|温习作用. 新手可以参考,也请高手指正错误, ...

  3. Entity Framework with MySQL 学习笔记一(查看EF和SQL请求日志)

    做数据库的一向来都会很注意请求的次数还有语句.这关系到很多性能的问题. 因此在使用EF的时候如果不了解原理很可能会搞出很糟糕的请求. 所以呢,在还没有搞懂EF前最基本的是你得"看得见&quo ...

  4. Entity Framework with MySQL 学习笔记一(乐观并发)

    在做项目时,通常我们对乐观并发有几种处理模式 1. 告诉用户此数据已被其他人捷足先登,更改了.你就算新一下重来吧. 2.直接把数据覆盖上去,我最大. 3.用被人的数据. 这里给出 code first ...

  5. Entity Framework with MySQL 学习笔记一(继承)

    基本上sql中要表示继承关系有3中方式. 分别是,1表继承(TPH),2表继承(TPC),3表继承(TPT) 1表 : Person id type name classroom office 1 s ...

  6. Entity Framework with MySQL 学习笔记一(insert,update,delete)

    先说说 insert 吧. 当EF执行insert时,如果我们传入的对象是有关联(1对多等)的话,它会执行多个语句 insert到多个表, 并且再select出来填充我们的属性(因为有些column默 ...

  7. Entity Framework with MySQL 学习笔记一(拦截)

    参考 : http://msdn.microsoft.com/en-us/data/dn469464.aspx EF 允许我们在发送SQL请求和返回数据时做一些拦截的动作 比如可以自定义写 log , ...

  8. Entity Framework with MySQL 学习笔记一(验证标签)

    直接上代码 [Table("single_table")] public class SingleTable { [Key] public Int32 id { get; set; ...

  9. Entity Framework with MySQL 学习笔记一(复杂类型 Complex Types)

    有时候我们希望在sql一个表里面的column, 一部分被分化成另一个class 典型的例子是 Address 直接看代码: [Table("member")] public cl ...

随机推荐

  1. 在SQL中使用自定义函数

      由于数据库的一个表字段中多包含html标签,现在需要修改数据库的字段把html标签都替换掉.当然我可以通过写一个程序去修改,那毕竟有点麻烦.直接在查询分析器中执行,但是MS SQL Server并 ...

  2. Markdown入门指南-指间阁

    宗旨 Markdown 的目标是实现「易读易写」. 可读性,无论如何,都是最重要的.一份使用 Markdown 格式撰写的文件应该可以直接以纯文本发布,并且看起来不会像是由许多标签或是格式指令所构成. ...

  3. nagios 实现Mysql 主从同步状态的监控

    一.系统环境 主机名 IP nagios 192.168.15.111 mysql_s 192.168.15.21 二.操作步骤 2.1 mysql_s端的配置 2.1.1 编写check_mysql ...

  4. js怪招(摘录篇)

    利用a标签自动解析URL 很多时候我们有从一个URL中提取域名,查询关键字,变量参数值等的需要,而万万没想到可以让浏览器方便地帮我们完成这一任务而不用我们写正则去抓取.方法就在JS代码里先创建一个a标 ...

  5. Android开发系列之button事件的4种写法

    经过前两篇blog的铺垫,我们今天热身一下,做个简单的样例. 文件夹结构还是引用上篇blog的截图. 详细实现代码: public class MainActivity extends Activit ...

  6. Java基础知识强化91:DateFormat类之DateFormat实现日期和字符串的相互转换

    1. DateFormat类概述: DateFormat 是日期/时间格式化子类的抽象类,它以与语言无关的方式格式化并解析日期或时间. 是抽象类,所以使用其子类SimpleDateFormat 2.  ...

  7. [CSAPP笔记][第九章虚拟存储器][吐血1500行]

    9.虚拟存储器 为了更加有效地管理存储器且少出错,现代系统提供了对主存的抽象概念,叫做虚拟存储器(VM). 虚拟存储器是硬件异常,硬件地址翻译,主存,磁盘文件和内核软件的完美交互. 为每个进程提供一个 ...

  8. ASP.NET-FineUI开发实践-2

    FineUI好处之一在于No JS,这里的No JS并不是不使用JS,JS对于ASP.Net是必不可少的,只是FineUI把大部分JS封装,如果想用,后台提供了很多方法返回JS,Get...Refer ...

  9. HTML基础总结<段落>

    HTML 段落 段落是通过 <p> 标签定义的. 实例 <p>This is a paragraph </p><p>This is another pa ...

  10. 排序算法之奇偶排序 JAVA奇偶排序算法

    奇偶排序法的思路是在数组中重复两趟扫描.第一趟扫描选择所有的数据项对,a[j]和a[j+1],j是奇数(j=1, 3, 5……).如果它们的关键字的值次序颠倒,就交换它们.第二趟扫描对所有的偶数数据项 ...