EF关系规则
参考文章:http://www.cnblogs.com/feigao/p/4617442.html
Entity Framework 实体间的关系,一对一,一对多,多对多,根据方向性来说又分为双向和单向。Code First在实体关系上有以下约定
1. 两个实体,如果一个实体包含一个引用属性,另一个实体包含一个集合属性,Code First默认约定它们为一对多关系。
2. 两个实体,如果只有一个实体包含一个导航属性或一个集合属性,Code First也默认约定它们是一对多关系。
3. 两个实体分别包含一个集合属性,Code First默认约定它们为多对多关系。
4. 两个实体分别包含一个引用属性,Code First默认约定它们为一对一关系。
5. 在一对一关系情况下,需要提供给Code First额外的信息,以确定它们的主从关系。
6. 在实体中定义一个外键属性,Code First使用属性是否为空来确定关系是必须还是可选
一对一:
以产品和保修卡为例,一个产品对应一个保修卡,产品和保修卡使用相同的产品编号
产品类
public class Product { public int Id { get; set; } /// <summary> /// 产品名字 /// </summary> public string Name { get; set; } /// <summary> /// 描述 /// </summary> public string Description { get; set; } /// <summary> /// 保修卡 /// </summary> public virtual WarrantyCard WarrantyCard { get; set; } }
保修卡类:
public class WarrantyCard { public int ProductId { get; set; } public DateTime ExpiredDate { get; set; } public virtual Product Product { get; set; } }
public class WarrantyCardMap : EntityTypeConfiguration<WarrantyCard>, IEntityMapper { public WarrantyCardMap() { ToTable("WarrantyCard"); HasKey(i => i.ProductId); } public void RegistTo(System.Data.Entity.ModelConfiguration.Configuration.ConfigurationRegistrar configurations) { configurations.Add(this); } }
public class ProductMap : EntityTypeConfiguration<Product>, IEntityMapper { public ProductMap() { ToTable("Product"); HasKey(p => p.Id); HasRequired(p => p.WarrantyCard).WithRequiredPrincipal(i => i.Product); HasOptional(p => p.WarrantyCard).WithRequired(i => i.Product); } }
Product作为Principal(主)而WarrantyCard作为Dependent
一对一(可空):
以产品和发票为例,有些产品可能没有发票
发票类:
public class Invoice { public int Id { get; set; } public string InvoiceNo { get; set; } public DateTime CreateDate { get; set; } }
产品类新增
public virtual Invoice Invoice { get; set; } public int? InvoiceId { get; set; }
在ProductMap中配置
HasOptional(p => p.Invoice).WithMany().HasForeignKey(p => p.InvoiceId);
HasOptional表示可空,WithMany()中没有参数标识不需要从发票关联到产品
一对多:
以产品和照片为例:
产品照片类:
public class ProductPhoto { public int Id { get; set; } public string FileName { get; set; } public float FileSize { get; set; } public virtual Product Product { get; set; } public int ProductId { get; set; } }
Product类增加ProductPhoto集合:
public virtual ICollection<ProductPhoto> Photos { get; set; }
ProductMap中配置
HasMany(p => p.Photos).WithRequired(pp => pp.Product).HasForeignKey(pp => pp.ProductId);
public class ProductPhotoMap : EntityTypeConfiguration<ProductPhoto>,IEntityMapper { public ProductPhotoMap() { ToTable("ProductPhoto"); HasKey(pp => pp.Id); } public void RegistTo(System.Data.Entity.ModelConfiguration.Configuration.ConfigurationRegistrar configurations) { configurations.Add(this); } }
HasMany表示Product中有多个ProductPhoto,WithRequired表示ProductPhoto一定会关联到一个Product
多对多:
产品和产品标签,一个产品可以有多个标签,一个标签也可对应多个产品
标签类:
public class Tag { public int Id { get; set; } public string Text { get; set; } public virtual ICollection<Product> Products { get; set; } }
Product增加标签集合
/// <summary> /// 标签 /// </summary> public virtual ICollection<Tag> Tags { get; set; }
ProductMap中添加配置:
HasMany(p => p.Tags).WithMany(t => t.Products).Map(m => m.ToTable("Product_Tag_Mapping"));
public class TagMap : EntityTypeConfiguration<Tag>,IEntityMapper { public TagMap() { ToTable("Tag"); HasKey(t => t.Id); } public void RegistTo(System.Data.Entity.ModelConfiguration.Configuration.ConfigurationRegistrar configurations) { configurations.Add(this); } }