ASP.NET Core使用EF创建模型(索引、备用键、继承、支持字段)

 更新时间:2022年04月07日 17:13:15   作者:暗断肠  
这篇文章介绍了ASP.NET Core使用EF创建模型的的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

1.索引

索引是跨多个数据存储区的常见概念。尽管它们在数据存储中的实现可能会有所不同,但也可用于基于列(或一组列)更高效地进行查找。

1.1约定

按照约定,将在用作外键的每个属性(或一组属性)中创建索引。

1.2数据批注

不能使用数据批注创建索引。

1.3Fluent API

您可以使用熟知的API来指定单个属性的索引。默认情况下,索引不是唯一的。

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Blog>()
            //配置索引
            .HasIndex(b => b.Url);
    }
}
public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }
}

您还可以指定索引应是唯一的,这意味着对于给定的属性,不能有两个实体具有相同的值。

modelBuilder.Entity<Blog>()
.HasIndex(b => b.Url)
.IsUnique();

您还可以为多个列指定索引。

class MyContext : DbContext
{
    public DbSet<Person> People { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Person>()
            .HasIndex(p => new { p.FirstName, p.LastName });
    }
}
public class Person
{
    public int PersonId { get; set; }
    public string FirstName { get; set; }
    public string LastName { get; set; }
}

2.备用键

备用键与主键相对,用作每个实体实例的备用唯一标识符。备用键可用作关系的目标。使用关系数据库时,这将映射到备用键列上的唯一索引/约束和引用列的一个或多个外键约束的概念。系统通常会在需要时为你引入备用键,你无需手动配置它们。

2.1约定

按照约定,系统将在识别属性(不是主键)时为你引入备用键,充当关系的目标。

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }
    public DbSet<Post> Posts { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Post>()
            .HasOne(p => p.Blog)
            .WithMany(b => b.Posts)
            .HasForeignKey(p => p.BlogUrl)
            .HasPrincipalKey(b => b.Url);
    }
}
public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }
    
    public List<Post> Posts { get; set; }
}
public class Post
{
    public int PostId { get; set; }
    public string Title { get; set; }
    public string Content { get; set; }
    //备用键
    public string BlogUrl { get; set; }
    public Blog Blog { get; set; }
}

2.2数据注释

不能使用数据注释配置备用键。

2.3Fluent API

你可以使用熟知的API将单个属性配置为备用密钥。

class MyContext : DbContext
{
    public DbSet<Car> Cars { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Car>()
            //配置为备用密钥
            .HasAlternateKey(c => c.LicensePlate);
    }
}
class Car
{
    public int CarId { get; set; }
    public string LicensePlate { get; set; }
    public string Make { get; set; }
    public string Model { get; set; }
}
你还可以使用熟知的API将多个属性配置为备用密钥(称为复合备用键)。
class MyContext : DbContext
{
    public DbSet<Car> Cars { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Car>()
            //配置为备用密钥
            .HasAlternateKey(c => new { c.State, c.LicensePlate });
    }
}
class Car
{
    public int CarId { get; set; }
    public string State { get; set; }
    public string LicensePlate { get; set; }
    public string Make { get; set; }
    public string Model { get; set; }
}

3.继承

EF模型中的继承用于控制如何在数据库中表示实体类中的继承。

3.1约定

按照约定,由数据库提供商确定如何在数据库中表示继承。有关如何使用关系数据库提供程序来处理此情况的详细说明。如果模型中显式包括两个或更多个继承类型,则EF仅会设置继承。EF不会扫描模型中未包含的基类型或派生类型。可以通过为继承层次结构中的每个类型公开DbSet,在模型中包含类型。

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }
    public DbSet<RssBlog> RssBlogs { get; set; }
}
public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }
}
public class RssBlog : Blog
{
    public string RssUrl { get; set; }
}

如果不想公开层次结构中一个或多个实体的DbSet,可以使用熟知的API来确保它们包含在模型中。如果不依赖约定,则可以使用HasBaseType显式指定基类型。

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<RssBlog>().HasBaseType<Blog>();
    }
}

4.支持字段

支持字段允许EF读取和写入字段,而不是属性。当使用类中的封装来限制或通过应用程序代码对数据访问进行限制时,这可能很有用,但在不使用这些限制的情况下,应从数据库中读取或写入值。

4.1约定

按照约定,将发现以下字段作为给定属性的支持字段(按优先级顺序列出)。仅为模型中包含的属性发现字段。

public class Blog
{
    private string _url;
    public int BlogId { get; set; }
    public string Url
    {
        get { return _url; }
        set { _url = value; }
    }
}

配置了支持字段后,当从数据库具体化实体实例(而不是使用属性资源库)时,EF将直接写入该字段。如果EF需要在其他时间读取或写入值,则它将使用属性(如果可能)。例如,如果EF需要更新某个属性的值,则它将使用属性setter(如果已定义)。如果该属性为只读,则它将写入字段。

4.2数据注释

不能通过数据批注配置支持字段。

4.3Fluent API

你可以使用熟知的API来配置属性的支持字段。

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Blog>()
            .Property(b => b.Url)
            .HasField("_validatedUrl");
    }
}
public class Blog
{
    private string _validatedUrl;
    public int BlogId { get; set; }
    public string Url
    {
        get { return _validatedUrl; }
    }
    public void SetUrl(string url)
    {
        using (var client = new HttpClient())
        {
            var response = client.GetAsync(url).Result;
            response.EnsureSuccessStatusCode();
        }
        _validatedUrl = url;
    }
}

4.3.1控制何时使用字段

可以配置EF何时使用字段或属性。有关支持的选项,请参阅PropertyAccessMode枚举。

modelBuilder.Entity<Blog>()
    .Property(b => b.Url)
    .HasField("_validatedUrl")
    .UsePropertyAccessMode(PropertyAccessMode.Field);

4.3.2没有属性的字段

你还可以在你的模型中创建一个概念属性,该属性在实体类中不具有相应的CLR属性,而是使用字段来存储实体中的数据。这不同于阴影属性,其中的数据存储在更改跟踪器中。如果实体类使用方法获取或设置值,通常会使用此方法。可以在Property(...) API中为EF指定字段的名称。如果没有具有给定名称的属性,则EF将查找字段。

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Blog>()
            .Property("_validatedUrl");
    }
}
public class Blog
{
    private string _validatedUrl;
    public int BlogId { get; set; }
    public string GetUrl()
    {
        return _validatedUrl;
    }
    public void SetUrl(string url)
    {
        using (var client = new HttpClient())
        {
            var response = client.GetAsync(url).Result;
            response.EnsureSuccessStatusCode();
        }
        _validatedUrl = url;
    }
}

您还可以选择为属性指定名称,而不是字段名称。然后,在创建模型时使用此名称,最值得注意的是,该名称将用于在数据库中映射到的列名称。

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Blog>()
        .Property<string>("Url")
        .HasField("_validatedUrl");
}

如果实体类中没有属性,则可以在LINQ查询中使用EF.Property(...)方法来引用概念上是模型的一部分的属性。

var blogs = db.blogs.OrderBy(b => EF.Property<string>(b, "Url"));

到此这篇关于ASP.NET Core使用EF创建模型的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • asp.net 分页显示数据表的数据的代码

    asp.net 分页显示数据表的数据的代码

    asp.net显示第一页、上一页、下一页和最后一页的分页显示数据表的数据
    2010-03-03
  • 详解.net core webapi 前后端开发分离后的配置和部署

    详解.net core webapi 前后端开发分离后的配置和部署

    这篇文章主要介绍了.net core webapi 前后端开发分离后的配置和部署,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-04-04
  • .NET装饰模式讲解

    .NET装饰模式讲解

    这篇文章主要为大家详细介绍了ASP.NET装饰模式的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-11-11
  • .net 数据表格显示控件介绍

    .net 数据表格显示控件介绍

    这篇文章主要介绍了.net 数据表格显示控件有哪些,适用于哪些场景,需要的朋友可以参考下
    2014-06-06
  • ASP.NET Core使用EF创建模型(包含属性、排除属性、主键和生成值)

    ASP.NET Core使用EF创建模型(包含属性、排除属性、主键和生成值)

    这篇文章介绍了ASP.NET Core使用EF创建模型的的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-04-04
  • ASP.net如何连接SQL SERVER 2012数据库

    ASP.net如何连接SQL SERVER 2012数据库

    这篇文章主要介绍了ASP.net连接SQL SERVER 2012数据库的方法,非常不错,在项目开发中经常可以用到,需要的朋友可以参考下
    2016-08-08
  • .Net Core2.1 WebAPI新增Swagger插件详解

    .Net Core2.1 WebAPI新增Swagger插件详解

    这篇文章主要给大家介绍了关于.Net Core2.1 WebAPI新增Swagger插件的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-07-07
  • docker部署Asp.net core应用的完整步骤

    docker部署Asp.net core应用的完整步骤

    这篇文章主要给大家介绍了关于docker部署Asp.net core应用的完整步骤,文中通过示例代码介绍的非常详细,对大家学习或者使用Asp.net core具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-05-05
  • asp.net 临时数据保存实现代码

    asp.net 临时数据保存实现代码

    在一个程序运行的时候往往我们会回到上一个页面,或者我们会需要当时留在这个页面的临时数据,例如,我们用百度搜索“脚本”之后我们会看到很多那啥我就不再说了啊!然后我们返回本来页面在文本框内会有“脚本”的字样,我们一下代码段就是实现这个功能
    2012-04-04
  • ashx文件的使用小结

    ashx文件的使用小结

    这篇文章主要是对ashx文件的使用进行了介绍。需要的朋友可以过来参考下,希望对大家有所帮助
    2013-12-12

最新评论