一.查询类型

  此功能是EF Core 2.1中的新功能。 EF Core除了实体类型之外,EF Core模型还可以包含查询类型,这些查询类型是针对“未映射到实体类型”的数据获取。比如视图,或只读数据表。

  

  1.1 下面介绍下,查询类型与实体类型共同与不同

    (1) 可以在OnModelCreating中或通过派生DbContext上的“set”属性添加到EF Core模型中。

    (2) 支持许多相同的映射功能,在关系数据库上,如继承映射和导航属性。也可以配置目标数据库对象和列通过 fluent API 方法或数据注释。

    但是,它们不同于实体中的类型:

    (1) 不需要定义的键。

   (2) 不会跟踪DbContext上的更改,因此不会在数据库中插入、更新或删除。

    (3) 永远不会由约定发现。

    (4) 仅支持一部分导航映射功能

  它们可能永远不会作为关系的主体端(没有主体实体和依赖实体的关系)。

  它们仅可包含指向的实体的引用导航属性。

  实体不能包含查询类型的导航属性。

    (5) 在ModelBuilder上使用Query方法而不是Entity方法进行寻址。

    (6) 是否通过类型DbQuery<T>而不是DbSet<T>的属性映射到DbContext上。

    (7) 映射到使用的数据库对象ToView方法,而非ToTable。

    (8) 可以映射到定义查询。定义查询是在模型中声明的辅助查询,充当查询类型的数据。

  1.2 查询类型使用方案(应用场景)

    (1) 作为返回类型的即席FromSql()查询。

    (2) 映射到数据库视图。

    (3) 映射到不具有定义的主键的表。

    (4) 映射到模型中定义的查询。

  

  1.3 映射到数据库对象

    查询类型映射到的数据库对象,通过实现ToView fluent API此ToView方法中指定的数据库对象,从 EF Core 的角度来看是视图,这意味着它将被视为只读查询源和不能作为目标的更新、插入或删除操作(也可以将被视为只读表)。 相反,对于实体类型,EF Core 假定数据库对象中指定ToTable方法可以视为表或视图、 意味着它可以用作查询源,还可以做更新、 删除和插入操作。

      // 查询类型的Blog ,  参数: view or table.
modelBuilder.Query<Blog>().ToView("xx");
// 实体类型的Blog 参数: view or table。注意如果是视图,更新数据,只能是一个表的视图。
modelBuilder.Entity<Blog>().ToTable("xx");

  1.4 案例

    下面的示例演示如何使用查询类型来查询数据库视图。完整代码可查看官方示例(下面有链接)。

     (1)首先定义一个简单的Blog和Post实体类型。

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; }
}

    (2) 使用BloggingContext上下文生成一个简单的数据库视图,这样就可以查询与每个Blog的帖子(Posts)数

          BloggingContext.Database.ExecuteSqlCommand(
@"CREATE VIEW View_BlogPostCounts AS
SELECT b.Name, Count(p.PostId) as PostCount
FROM Blogs b
JOIN Posts p on p.BlogId = b.BlogId
GROUP BY b.Name");

          

    (3) 接下来,我们定义一个查询类来保存数据库视图的结果

public class BlogPostsCount
{
public string BlogName { get; set; }
public int PostCount { get; set; }
}

    (4) 我们配置中的查询类型在OnModelCreating中使用modelBuilder.Query<T>API。 我们使用标准的 fluent 配置 Api 来配置查询类型的映射:

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder
.Query<BlogPostsCount>().ToView("View_BlogPostCounts")
.Property(v => v.BlogName).HasColumnName("Name");
}

    (5) 最后,我们可以采用标准方式来查询数据库视图:

  var postCounts = BloggingContext.BlogPostCounts.ToList();

    

二. 关系数据库建模

  一般而言,本部分中的配置适用于关系数据库。安装关系数据库提供程序时,下面显示的扩展方法将变为可用,原因在于共享Microsoft.EntityFrameworkCore.Relational包

  

  2.1 表映射

    表映射是指(实体与数据表的映射)包括查询的表数据,并将其保存到数据库中。

    按照约定,每个实体都将映射到一个表,该表的名称与DbSet<TEntity>在派生上下文中公开实体的属性相同。例如在EF上下文中公开Blog类型,生成的表名与属性名相同。

    //生成Blogs表名
public DbSet<Blog> Blogs { get; set; }
    //除了约定还可以使用数据注释,表blogs与实体Blog映射
[Table("blogs")]
public class Blog
{
public int BlogId { get; set; }
public string Url { get; set; }
} //还可以使用Fluent API 配置,功能实现与上面一样
modelBuilder.Entity<Blog>().ToTable("blogs");

  

  2.2 列映射

    按照约定,实体中的每个属性都会映射到表中,具有相同名称的数据字段。还可以使用数据注释或Fluent API 配置:

public class Blog
{
//数据注释将表Blog的blog_id字段映射到BlogId属性。
[Column("blog_id")]
public int BlogId { get; set; }
public string Url { get; set; }
} //使用Fluent API配置,功能实现与上面一样
modelBuilder.Entity<Blog>().Property(b => b.BlogId).HasColumnName("blog_id");

  2.3 数据类型

    数据类型是指:数据库的数据类型与 CLR 属性类型映射。可以使用数据注释来指定精确的数据类型的列(一般用在code first模式)。

//数据注释的案例
public class Blog
{
public int BlogId { get; set; }
[Column(TypeName = "varchar(200)")]
public string Url { get; set; }
[Column(TypeName = "decimal(5, 2)")]
public decimal Rating { get; set; }
}
//Fluent API配置,功能实现与上面一样
modelBuilder.Entity<Blog>(eb =>
{
eb.Property(b => b.Url).HasColumnType("varchar(200)");
eb.Property(b => b.Rating).HasColumnType("decimal(5, 2)");
});

  2.4 主键

    对于每个实体类型的键是引入了主键约束。按照约定,主键名称是PK_<type name>,例如Posts 实体中有BlogId属性,那表中主键约束名是PK_Posts。可以使用Fluent API配置数据库中的主键约束的名称。

  //将默认的PK_Posts改为了PrimaryKey_BlogId主键约束名
modelBuilder.Entity<Blog>()
.HasKey(b => b.BlogId)
.HasName("PrimaryKey_BlogId");

  2.5 默认架构

    如果对象没有显式配置架构,则默认架构是创建对象的数据库架构。按照约定,数据库提供程序将选择最合适的默认架构。例如,Microsoft SQL Server将使用该dbo模式。无法使用数据注释设置默认架构,可以使用Fluent API指定默认架构。

    -- 修改sqlserver 表默认架构
    ALTER SCHEMA [blogging] TRANSFER dbo.Blogs
    -- 查询表
    select * from [blogging].Blogs
      //设置ef core 对应数据库表查询的架构名
  modelBuilder.HasDefaultSchema("blogging");

  2.6 默认值

    如果插入新行,但未指定列值,可设置一个默认值。不可以使用约定或数据注释提供默认值。可以使用Fluent API 配置默认值.

    //例如插入数据时,设置Created字段取sqlserver的当前时间值。
modelBuilder.Entity<Blog>()
.Property(b => b.Created)
.HasDefaultValueSql("getdate()");

  2.7 索引

    关系数据库中的索引映射到相同的概念的 EF core 中的索引。按照约定,索引名为IX_<type name>_<property name>对于复合索引,下划线分隔的属性名称列表。可以使用Fluent API配置索引的名称。

      //设置url索引名称为Index_Url
modelBuilder.Entity<Blog>()
.HasIndex(b => b.Url)
.HasName("Index_Url"); //还可以指定索引过滤器
modelBuilder.Entity<Blog>()
.HasIndex(b => b.Url)
.HasFilter("[Url] IS NOT NULL");

  参考文献:

    官方文档:EF查询类型

           查询类型示例

          关系数据库建模

asp.net core系列 29 EF模型配置(查询类型,关系数据库建模)的更多相关文章

  1. asp.net core系列 23 EF模型配置(概述, 类型和属性的包含与排除)

    一.模型配置概述 EF使用一组约定基于实体类的定义来构建模型. 可指定其他配置以补充或替代约定的内容.本系列介绍的配置可应用于面向任何数据存储的模型,以及面向任意关系数据库时可应用的配置. 数据库提供 ...

  2. asp.net core系列 28 EF模型配置(字段,构造函数,拥有实体类型)

    一. 支持字段 EF允许读取或写入字段而不是一个属性.在使用实体类时,用面向对象的封装来限制或增强应用程序代码对数据访问的语义时,这可能很有用.无法使用数据注释配置.除了约定,还可以使用Fluent ...

  3. asp.net core系列 27 EF模型配置(索引,备用键,继承)

    一.索引 索引是许多数据存储中的常见概念.虽然它们在数据存储中的实现可能会有所不同,但它们可用于更有效地基于列(或列集)进行查找.按照约定,用作外键每个属性 (或组的属性) 会自动创建索引.无法使用数 ...

  4. asp.net core系列 25 EF模型配置(隐藏属性)

    一. 隐藏属性概述 隐藏属性也叫影子属性,该属性不是在.net实体类中定义的属性,而是在EFCore模型中为该实体类型定义的属性.这些属性的值和状态完全在变更跟踪器中维护.它有二个功能:(1)当数据库 ...

  5. asp.net core系列 24 EF模型配置(主键,生成值,最大长度,并发标记)

    一.主键 键用作每个实体实例的主要唯一标识符. 使用关系数据库时,这会映射到主键的概念. 还可以配置不是主键的唯一标识符.按照约定,名为 Id 或 <type name>Id 的属性会配置 ...

  6. asp.net core系列 26 EF模型配置(实体关系)

    一.概述 EF实体关系定义了两个实体互相关联起来(主体实体和依赖实体的关系,对应数据库中主表和子表关系). 在关系型数据库中,这种表示是通过外键约束来体现.本篇主要讲一对多的关系.先了解下描述关系的术 ...

  7. asp.net core系列 30 EF管理数据库架构--必备知识 迁移

    一.管理数据库架构概述 EF Core 提供两种主要方法来保持 EF Core 模型和数据库架构同步.一是以 EF Core 模型为基准,二是以数据库为基准. (1)如果希望以 EF Core 模型为 ...

  8. asp.net core系列 31 EF管理数据库架构--必备知识 反向工程

    一.   反向工程 反向工程是基于数据库架构,生成的实体类和DbContext类代码的过程,对于Visual Studio开发,建议使用PMC.对于其他开发环境,请选择.NET Core CLI工具( ...

  9. asp.net core 系列 22 EF(连接字符串,连接复原,DbContext)

    一.连接字符串 在上二篇中,ASP.NET Core 应用程序连接字符串是写死在ConfigureServices代码中,下面介绍通过配置来实现.连接字符串可以存储在 appsettings.json ...

随机推荐

  1. Redis之实战篇(与Mybatis整合)

    摘要: 现在市面流行的java框架,一个是ssh(spring.struts2.hibernate),另一个是ssm(spring.springmvc.mybatis),由于之前已经有整合过ssm框架 ...

  2. c语言之单链表的创建及排序

    今天对之前学习过的链表知识进行简单的总结顺便写点代码:创建一个链表有头插法跟尾插法两种,在下面代码中我们为结点分配的内存实在堆上分配的,因此需要我们手动释放,释放用free()函数 下面代码贴出具体代 ...

  3. JQ无缝轮播图-插件封装

    类似京东的这种无缝轮播效果: 实例代码下载 HTML代码: <body> <!-- /*觅me 探索生活*/ --> <div class="test" ...

  4. form 表单提交返回值问题

    form不比ajax,即使后台返回值后,在页面也不知道怎么去取值判断提交状态.所以前几天结合网上资料整了一个小案例,需要用到ajaxSubmit,即通过ajax来提交表单,好处在于可以在任何情况下进行 ...

  5. 一步一步实现HTTP服务器-开篇

    缘起 翻开清单,一条条计划一直列在那里,一天又一天,不知道什么时候写下了它,也知不道什么时候完成它,它一直在那静静的等待着. 静下心来,反思自己,才发现自己是多么的无知,多么的没有毅力.设定了无数目标 ...

  6. python 模块与包

    一.模块 1.1 什么是模块 # 什么是模块 # 一组功能的集合,就是模块,在python中一个py文件就一个模块 1.2 为什么要使用模块 # 1.可使代码的结构功能分区更清晰 # 2.可重复使用模 ...

  7. 一个bug分析 ----------换个角度,有另外一个天地

    有个接口是按修改时间(updated_time)排序的 优化后,有人反馈接口的返回值有问题 查了一下,反馈的数据是推荐过的(推荐操作是会更新updated_time的). 然后就认为是有人进行了推荐操 ...

  8. websocket(二)--简单实现网页版群聊

    websocket可以实现服务端的消息推送,而不必在客户端轮询,大大的节省的资源,对于实时通讯来说简直是个大喜讯. 在上一篇文章中介绍了协议握手,这篇文章将通过实现简单的群聊来帮助进一步了解webso ...

  9. 解析PHP程序员需要掌握的必备技能

    转自:http://www.php100.com/html/php/lei/2013/0904/4199.html 作为PHP的爱好者,如果你想加入PHP程序的世界,一定要做好充分的准备.建议大家阅读 ...

  10. 自动化测试用例getText()获取某一个元素的值返回null或空

    问题描述 页面上一个元素无法获取它的值. 用getText() 获取该元素的值返回null或空. 根本原因 ? 需要更多了解dom结构 解决办法 用getAttribute("value&q ...