Entity Framework映射TPH、TPT、TPC與繼承類
一、TPH
Table Per Hierarchy (默認,每個層次一個表)
每個層次結構共用一個表,類的每一個屬性都必須是可空的。
1、默認行為
隻建立一個表,把基類和子類中的所有屬性都映射為表中的列。
為基類和所有子類共建立一個表,基類和子類中的所有屬性都映射為表中的一個列。
默認在這個表中建立一個叫做Discriminator的列,類型是nvarchar,長度是128。在存儲基類或子類的時候,把類名作為Discriminator列的值。
2、Fluent API修改默認行為
Map方法中傳入的類型參數是子類的類名,Requires用於指定Discriminator列的名字,HasValue用於指定它的類型和每個子類對應的值。
modelBuilder.Entity<Course>() .Map<Course>(m => m.Requires("Type").HasValue("Course")) .Map<OnsiteCourse>(m => m.Requires("Type").HasValue("OnsiteCourse"));
二、TPT
Table Per Type(每個類各一個表)
1、默認行為
為基類和每個子類各建立一個表,每個與子類對應的表中隻包含子類特有的屬性對應的列。
子類的表中隻包含子類特有的屬性,子表還會存儲一個將子表與基表聯接的外鍵。
2、Fluent API修改默認行為
我們可以使用Map方法強制讓Code First使用TPT方式,因為Code First默認使用的是TPH方式。
modelBuilder.Entity<Course>().ToTable("Course"); modelBuilder.Entity<OnsiteCourse>().ToTable("OnsiteCourse");
三、TPC
Table Per ConCrete Type(每個具體類型各一個表)
每個具體的派生類各一個表,沒有基表。不推薦使用。
1、默認行為
在子類對應的表中除瞭子類特有的屬性外還有基類的屬性對應的表。基類可以是abstract。
2、Fluent API修改默認行為
通過MapInheritedProperties方法就可以強制Code First使用TPC方式。
註意:因為屬於 TPC 繼承層次結構的表並不使用同一個主鍵, 關閉主鍵屬性的標識,避免為不同子表插入重復的實體鍵。
modelBuilder.Entity<Course>() .Property(c => c.CourseID) .HasDatabaseGeneratedOption(DatabaseGeneratedOption.None); modelBuilder.Entity<OnsiteCourse>().Map(m => { m.MapInheritedProperties(); m.ToTable("OnsiteCourse"); }); modelBuilder.Entity<OnlineCourse>().Map(m => { m.MapInheritedProperties(); m.ToTable("OnlineCourse"); });
四、實體拆分
允許一個實體類型的屬性分散在多個表中。
實體拆分通過多次調用 Map 方法將一部分屬性映射到特定表。
在以下示例中,Department 實體拆分到兩個表中:Department 和 DepartmentDetails。
modelBuilder.Entity<Department>() .Map(m => { m.Properties(t => new { t.DepartmentID, t.Name }); m.ToTable("Department"); }). Map(m => { m.Properties(t => new { t.DepartmentID, t.Administrator, t.StartDate, t.Budget }); m.ToTable("DepartmentDetails"); });
五、表拆分
兩個實體類型映射到同一個表。
1.兩個類必須共享同一個主鍵。
2.兩個類之間的關系必須被映射為表之間的一對一關系。
modelBuilder.Entity<OfficeAssignment>().HasKey(t => t.InstructorID); //共用主鍵 modelBuilder.Entity<Instructor>() .HasRequired(t => t.OfficeAssignment).WithRequiredPrincipal(t => t.Instructor);//一對一關系 modelBuilder.Entity<Instructor>().ToTable("Instructor"); modelBuilder.Entity<OfficeAssignment>().ToTable("Instructor");
六、將類指定為復雜類型
1、指定方法:
DataAnnotations方式:
[ConlexType] public Details details;
或FluentAPI:
modelBuilder.ComplexType<Details>();
註意:
1.復雜類型類不能有主鍵。
2.復雜類型隻能包含.net基礎類型的屬性。
3.使用復雜類型的類,隻能包復雜類型的一個實例,不能使用復雜類型的集合。
2、配置復雜類型的屬性
(1)、可以對 ComplexTypeConfiguration 調用 Property。
modelBuilder.ComplexType<Details>() .Property(t => t.Location).HasMaxLength(20);
或
public class AddressComplexTypeConfiguration:ComplexTypeConfiguration<Address> { public AddressComplexTypeConfiguration() { Property(a => a.Country).HasColumnName("Country").HasMaxLength(100); Property(a => a.ZipCode).HasColumnName("ZipCode").HasMaxLength(6); } }
(2)、也可以使用點表示法訪問復雜類型的屬性。
modelBuilder.Entity<OnsiteCourse>().Property(t => t.Details.Location) .HasMaxLength(20);
七、DataBase初始化
1、調用Database.SetInitializer方法:
一般Global.ascx.cs,Main應用程序的入口等地方調用Database.SetInitializer方法:
- 隻要Fluent API配置的數據庫映射發生變化或者domain中的model發生變化瞭,就把以前的數據庫刪除掉,根據新的配置重新建立數據庫。
Database.SetInitializer(new DropCreateDatabaseIfModelChanges<BreakAwayContext>());
- 隻有在沒有數據庫的時候才會根據數據庫連接配置創建新的數據庫。這種配置主要用於production環境。
Database.SetInitializer(new CreateDatabaseIfNotExists<BreakAwayContext>());
- 不管數據庫映射或者model是否發生變化,每次都重新刪除並根據配置重建數據庫。
Database.SetInitializer(new DropCreateDatabaseAlways<BreakAwayContext>());
2、通過配置文件更靈活的指定數據庫初始化的方式:
<?xml version="1.0"?> <configuration> <appSettings> <add key="DatabaseInitializerForTypeOrderSystemContext" value="System.Data.Entity.DropCreateDatabaseIfModelChanges[[OrderSystemContext]], EntityFramework" /> </appSettings> </configuration>
3、自定義數據庫初始化類
通過自定的初始化類,還可以將一些基礎數據在創建數據庫之後插入到數據庫中去。
public class DropCreateOrderDatabaseWithSeedValueAlways : DropCreateDatabaseAlways<OrderSystemContext> { protected override void Seed(OrderSystemContext context) { context.ProductCatalogs.Add(new ProductCatalog { CatalogName = "DELL E6400", Manufactory = "DELL", ListPrice = 5600, NetPrice = 4300 }); context.ProductCatalogs.Add(new ProductCatalog { CatalogName = "DELL E6420", Manufactory = "DELL", ListPrice = 7000, NetPrice = 5400 }); } } Database.SetInitializer(new DropCreateOrderDatabaseWithSeedValueAlways());
到此這篇關於Entity Framework映射TPH、TPT、TPC與繼承類的文章就介紹到這瞭。希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。
推薦閱讀:
- Entity Framework使用DbModelBuilder API創建表結構
- Entity Framework管理一對二實體關系
- Entity Framework使用Code First的實體繼承模式
- Entity Framework使用配置夥伴創建數據庫
- Entity Framework使用Fluent API配置案例