本章的第一个示例映射到一个表。第二个更复杂的示例展示了如何创建表之间的关系。新数据库是通过编程创建的,当然也可以先创建数据库,再使用代码访问已有的数据库。
1. 创建关系
下面开始创建模型。示例项目使用MenuCard和Menu类型定义了一对多关系。MenuCard包含Menu对象的列表。这个关系由List<Menu>类型的Menu属性定义:
public class MenuCard
{
public int MenuCardId { get; set; }
public string Title { get; set; }
public List<Menu> Menus { get; } = new List<Menu>();
public override string ToString() => Title;
}
也可以在另一个方向上访问关系,Menu可以使用MenuCard属性访问MenuCard。指定MenuCardId属性来定义一个外键关系:
public class Menu
{
public int MenuId { get; set; }
public string Text { get; set; }
public decimal Price { get; set; }
public int MenuCardId { get; set; }
public MenuCard MenuCard { get; set; }
public override string ToString() => Text;
}
到数据库的映射是通过MenusContext类实现的。这个类的定义类似于前面的上下文类型;它只包含两个属性,映射到两个对象类型:Menus和MenuCards属性:
public class MenusContext : DbContext
{
private const string ConnectionString =
@"server=(localdb)\MSSQLLocalDB;database=MenusCards;trusted_connection=true";
public DbSet<Menu> Menus { get; set; }
public DbSet<MenuCard> MenuCards { get; set; }
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
base.OnConfiguring(optionsBuilder);
optionsBuilder.UseSqlServer(ConnectionString);
}
}
在创建代码中修改一些部分是有益的。例如,Text和Title列的尺寸可以减小为NVARCHAR(MAX)中的值。另外,SQL Server定义了Money类型,它可用于Price列,模式名称可以在dbo中修改。Entity Framework提供了两个选项——数据注释和流利API——用于在代码中进行这些修改,如下面所述。
2. 数据注释
要影响生成的数据库,一个方法是给实体类型添加数据注释。默认情况下,表的名称来自于上下文的属性。因此要映射Menu类,应使用Menus表,因为映射Menu的DbSet属性名为Menus。有了数据注释,可以使用Table特性来改变表格。要改变模式名称,Table特性定义Schema特性。为了给字符串类型指定另一个长度,可以应用MaxLength特性:
[Table("MenuCards", Schema = "mc")]
public class MenuCard
{
public int MenuCardId { get; set; }
[MaxLength(120)]
public string Title { get; set; }
public List<Menu> Menus { get; } = new List<Menu>();
public override string ToString() => Title;
}
注意:
EF Core默认使用上下文中的属性名称映射到表格上,这不同于Entity Framwork。Entity Framework使用一个字典查找单复数名称。
在Menu类中,应用了Table和MaxLength特性。为了更改SQL类型,可以使用Column特性:
[Table("Menus",Schema = "mc")]
public class Menu
{
public int MenuId { get; set; }
[MaxLength(50)]
public string Text { get; set; }
[Column(TypeName = "Money")]
public decimal Price { get; set; }
public int MenuCardId { get; set; }
public MenuCard MenuCard { get; set; }
public override string ToString() => Text;
}
CREATE TABLE [mc].[MenuCards] (
[MenuCardId] INT IDENTITY (1, 1) NOT NULL,
[Title] NVARCHAR (120) NULL,
CONSTRAINT [PK_MenuCards] PRIMARY KEY CLUSTERED ([MenuCardId] ASC)
);
CREATE TABLE [mc].[Menus] (
[MenuId] INT IDENTITY (1, 1) NOT NULL,
[Text] NVARCHAR (50) NULL,
[Price] MONEY NOT NULL,
[MenuCardId] INT NOT NULL,
CONSTRAINT [PK_Menus] PRIMARY KEY CLUSTERED ([MenuId] ASC),
CONSTRAINT [FK_Menus_MenuCards_MenuCardId] FOREIGN KEY ([MenuCardId]) REFEREN