【Entity Framework】你知道如何处理无键实体吗
文章目录
- 【Entity Framework】你知道如何处理无键实体吗
一、概述
Entity Framework
除了常规实体类型外,EF Core
模型还可以包含无键实体类型,可用于对不包含键值的数据执行ef="/ShuJuKu.html" title=数据库>数据库查询。
二、定义无键实体类型
可按如下定义无键实体类型:
ef="/C#.html" title=c#>c#">[Keyless]
public class BlogPostsCount
{public string BlogName{get;set;}public int PostCount{get;set;}
}
- Fluent API
ef="/C#.html" title=c#>c#">protected override void OnModelCreating(ModelBuilder modelBuilder)
{modelBuilder.Entity<BlogPostsCount>().HasNoKey();
}
三、无键实体类型特征
无键实体类型支持与常规实体类型相同的多个映射功能,例如继承映射和导航属性。在关系存储上,它们可以通过Fluent API
方法或数据注释来配置目标ef="/ShuJuKu.html" title=数据库>数据库对象和列。
但是,它们不同于常规实体类型,因为他们:
-
不能定义键
-
永远不会对
DbContext
中的更改进行跟踪,因此不会对ef="/ShuJuKu.html" title=数据库>数据库进行插入,更新或删除这些操作 -
绝不会被约定发现
-
仅支持导航映射功能的子集,具体如下:
- 它们永远不能充当关系的主体端
- 它们可能没有指向从属实体的导航
- 它们只能包含指向常规实体的引用导航属性
- 实体不能包含无键实体类型的导航属性
-
需要配置
[Keyless]
数据注释或.HasNoKey()
方法调用 -
可以映射到定义查询。 定义查询是在模型中声明的查询,它充当无键实体类型的数据源
-
可以具有层次结构,但必须映射为 TPH
-
不能使用表拆分或实体拆分
四、无键实体使用场景
无键实体类型的一些主要使用场景包括:
- 充当SQL查询的返回类型
- 映射到不包含主键的ef="/ShuJuKu.html" title=数据库>数据库视图
- 映射到未定义主键的表
- 映射到模型中定义的查询
五、无键实体使用场景
可以使用 ToTable
或 ToView
Fluent API 将无键实体类型映射到ef="/ShuJuKu.html" title=数据库>数据库对象。 从 EF Core 的角度来看,此方法中指定的ef="/ShuJuKu.html" title=数据库>数据库对象是一个视图,这意味着它将被视为只读查询源,并且不能作为更新、插入或删除操作的目标。 但是,这并不意味着ef="/ShuJuKu.html" title=数据库>数据库对象实际上必须是ef="/ShuJuKu.html" title=数据库>数据库视图。 它也可以是将被视为只读的ef="/ShuJuKu.html" title=数据库>数据库表。 相反,对于常规实体类型,EF Core 假设 ToTable
方法中指定的ef="/ShuJuKu.html" title=数据库>数据库对象可以作为表,这意味着它可用作查询源,但也可作为更新、删除和插入操作的目标。 事实上,可以在 ToTable
中指定ef="/ShuJuKu.html" title=数据库>数据库视图的名称,只要该视图配置为可在ef="/ShuJuKu.html" title=数据库>数据库上更新,一切都应能正常运行。
六、无键使用示例
6.1 定义一个简单的Blog和Post模型:
ef="/C#.html" title=c#>c#">public class Blog
{public int BlogId { get; set; }public string Name { get; set; }public string Url { get; set; }public ICollection<Post> Posts { get; set; }
}public class Post
{public int PostId { get; set; }public string Title { get; set; }public string Content { get; set; }public int BlogId { get; set; }
}
6.2 定义一个简单ef="/ShuJuKu.html" title=数据库>数据库视图
ef="/C#.html" title=c#>c#">db.Database.ExecuteSqlRaw(@"CREATE VIEW View_BlogPostCounts ASSELECT b.Name, Count(p.PostId) as PostCountFROM Blogs bJOIN Posts p on p.BlogId = b.BlogIdGROUP BY b.Name");
6.3 定义一个类来保存ef="/ShuJuKu.html" title=数据库>数据库视图的结果
ef="/C#.html" title=c#>c#">public class BlogPostsCount
{public string BlogName { get; set; }public int PostCount { get; set; }
}
6.4 使用HasNoKey API在OnModelCreating中配置无键实体类型
ef="/C#.html" title=c#>c#">protected override void OnModelCreating(ModelBuilder modelBuilder)
{modelBuilder.Entity<BlogPostsCount>(eb =>{eb.HasNoKey();eb.ToView("View_BlogPostCounts");eb.Property(v => v.BlogName).HasColumnName("Name");});
}
6.5 配置DbContext
以包含DbSet<T>
ef="/C#.html" title=c#>c#">public DbSet<BlogPostsCount> BlogPostCounts { get; set; }
6.7 采用标准方式查询ef="/ShuJuKu.html" title=数据库>数据库视图
ef="/C#.html" title=c#>c#">var postCounts = db.BlogPostCounts.ToList();
foreach (var postCount in postCounts)
{Console.WriteLine($"{postCount.BlogName} has {postCount.PostCount} posts.");Console.WriteLine();
}
七、总结
无键主要应用在SQL、视图、存储过程的查询,在制作报表时,应用比较多。