首页 > 其他分享 >Entity Framework教程-数据注解 和 FlentAPI(Data Annotations)

Entity Framework教程-数据注解 和 FlentAPI(Data Annotations)

时间:2022-10-18 08:45:10浏览次数:74  
标签:set get FlentAPI Entity modelBuilder 设置 Data public

更新记录
转载请注明出处:
2022年10月18日 发布。
2022年10月10日 从笔记迁移到博客。

实体配置

两种实体配置方式对比

Data Annotation:把配置以特性(Annotation)的形式标注在实体类中。优点:简单;缺点:耦合。

Fluent API:把配置写到单独的配置类中。优点:解耦;缺点:复杂。

大部分功能重叠。可以混用,但是不建议混用。

数据注解(Data Annotations)

说明

模型验证可以保证数据的准确性

验证可以用于域模型,可以在任意一个地方定义验证条件

注解说明

通常EF会按约定自动匹配数据库中的表和字段的类型,但也可以自己进行设置

当默认的对象中的数据类型与数据库中的数据类型不一致时,可以使用注解

命名空间

using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using System.ComponentModel.Design;

所有注解:

https://docs.microsoft.com/en-us/dotnet/api/system.componentmodel.dataannotations?redirectedfrom=MSDN&view=net-5.0

注意:

数据注解并不属于EF 的一部分,所以它在其他的命名空间下

数据注解还可以用于ASP.NET 用于验证约束数据

.NET5注意

从.NET 5开始需要单独安装NuGet包支持

System.ComponentModel.DataAnnotations

使用方法

引入命名空间

using System.ComponentModel.DataAnnotations;

说明:

DataAnnotations包含一组验证数据的特性

DataAnnotations.Schema包含将实体类映射到数据库表的特性

然后声明验证规则(Declarative Validation Rules)

在类或者类的成员上方设置注解

public class GusetMessage
{
    [Required(ErrorMessage = "Must Input Your Name")]
    public virtual string Name { get; set; }
    [Required(ErrorMessage = "Must Input Your Mail")]
    public virtual string Mail { get; set; }
    [Required(ErrorMessage = "Must Input Your Phone")]
    public virtual string Phone { get; set; }
    [Required(ErrorMessage = "Must Input")]
    public virtual bool? WillAttend { get; set; }
}

常用预定义数据注解

指定数据库底层具体数据表

[Table("Product")]
class Product
{
}

指定表和架构

[Table("Product", Schema = "Store")]
class Product
{
}

指定数据库底层具体字段

class Product
{
    [Column("Name")]
    public string ProductName { get; set; }
}

同时指定字段的数据类型

class Product
{
    [Column("Name",TypeName = "VARCHAR(300)")]
    public string ProductName { get; set; }
}

字段指定主键

[Table("PandaProduct")]
class Product
{
    [Key]
    public int ProductId { get; set; }
}

设置主键自动生成

[Key]
[DatabaseGenerated(DatabaseGeneratedOption.Identity)]
public Guid Id { get; set; }

设置主键不自动生成

[DatabaseGenerated(DatabaseGeneratedOption.None)]

设置字段为数据库自动运算

设置字段为自动增长字段

[Key]
[DatabaseGenerated(DatabaseGeneratedOption.Identity)]
public int ProductId { get; set; }

设置字段为数据库提供的默认值

[DatabaseGenerated(DatabaseGeneratedOption.None)]
public string ProductName { get; set; }

字段指定外键

[ForeignKey("外键名称")]
public virtual DepartmentId { get; set;}

字段必须填写

[Required]
public string ProductType { get; set; }

指定字段的长度

设置最大长度

[MaxLength(500)]
public string ProductType { get; set; }

设置最大长度并设置提示文本

[MaxLength(500,ErrorMessage = "太大了")]
public string ProductType { get; set; }

设置最小长度
注意:如果是NULL或者空字符同样不符合条件

[MinLength(10)]
public string ProductType { get; set; }

设置最小长度并设置提示文本

[MinLength(10,ErrorMessage = "太小了")]
public string ProductType { get; set; }

设置字符串的最大长度

[StringLength(160)]
public string Name { get; set; }

设置字符串的最大长度和最小长度

[StringLength(160,MinimumLength = 3)]
public string Name { get; set; }

指定字符串字段长度

[StringLength(50, ErrorMessage = "The value of this field is limited to 50 characters")]

字段指定默认值

[DefaultValue(<the_value>)]
[DefaultValue(<the_value>)]

实例:

[DefaultValue(123)]
[DefaultValue(false)]

设置不用映射

设置字段不用映射
部分字段用于程序处理,比如用于运算,不需要映射到数据库之中

class Product
{
    [NotMapped]
    public string ProductType3 { get; set; }
}

设置模型不用映射
使用[NotMapped]注解修饰Model,Model将不会映射到数据库中

[NotMapped]
class Product
{
    [Key]
    public int ProductId { get; set; }
    [StringLength(200)]
    public string ProductName { get; set; }
}

正则表达式验证

正则表达式验证

[RegularExpression(@"RegularExpression")]
public string Note { get; set; }
[RegularExpression("^[a-zA-Z]+$")]
public string Name { get; set; }
[RegularExpression("^[a-zA-Z]+$", ErrorMessage = "Only allowed")]
public string FirstName { get; set; }

设置邮箱验证

注意:在代码层面验证,而不是在数据库层面,约束最终不会体现在数据库中

[EmailAddress(ErrorMessage = "Invalid Email Address")]

设置网址验证

注意:在代码层面验证,而不是在数据库层面,约束最终不会体现在数据库中

[Url]
public string Url { get; set; }
[Url(ErrorMessage = "Provide a valid url")]
public string Url { get; set; }

设置类型的范围

设置指定数字类型的范围

[Range(typeof(decimal),"0.0","166.66")]
public int Code { get; set; }

设置指定日期类型的范围

[Range(typeof(DateTime),"1/1/2000","12/30/2020",ErrorMessage = "错误信息")]
public DateTime CreateTime { get; set; }

设置字符串的长度范围

[Range(10,100)]
public string ProductType { get; set; }

设置范围并设置错误提示信息

[Range(1,200,ErrorMessage = "年龄必须在1~200内")]
public int Age { get; set; }

与其他属性一样的模型验证

[Range(typeof(decimal),"0.0","166.66")]
public int Code { get; set; }
[Compare("Code",ErrorMessage = "数据xxxx")]
public int Age { get; set; }

提示错误信息

每个数据验证注解都可以使用ErrorMessage属性来定义错误信息

[Required(ErrorMessage = "必须提供Code代码")]
public int Code { get; set; }
[StringLength(160,ErrorMessage = "数据的最大长度为160")]
public string Name { get; set; }

定义强类型视图渲染后显示的名称

[Display(Name = "编号")]
public int Code { get; set; }

还可以定义强类型视图渲染后显示的顺序(顺序从10000开始)

[Display(Name = "编号",Order = 10002)]
public int Code { get; set; }
[Display(Name = "姓名",Order = 10001)]
public string Name { get; set; }

设置渲染显示的HTML表单类型

[DataType(DataType.Password)]
public string Password { get; set; }

还可以是这些类型

[DataType(DataType.EmailAddress)]
[DataType(DataType.Password)]
[DataType(DataType.Currency)]
[DataType(DataType.Date)]
[DataType(DataType.DateTime)]
[DataType(DataType.Time)]
[DataType(DataType.Html)]
[DataType(DataType.ImageUrl)]
[DataType(DataType.Url)]
[DataType(DataType.MultilineText)]
[DataType(DataType.PhoneNumber)]
[DataType(DataType.PostalCode)]
[DataType(DataType.Text)]
[DataType(DataType.Upload)]
[DataType(DataType.Currency)]
public decimal Price { get; set; } = 5;

设置底层类型为货币

[DataType(DataType.Currency)]
public decimal Price { get; set; } = 5;

设置将字段渲染为

[HiddenInput]
public int Code { get; set; }

设置映射到数据库中的表字段名称

[Column("ProductName")]
public string ProductName { get; set; }

设置表的 字段名称 和 字段类型

[Column("ProductName",TypeName = "VARCHAR(20)")]

指定索引字段

[Index]
public string ProductType3 { get; set; }

指定索引名称

[Index("IX_PandaIndex")]
public string ProductType3 { get; set; }

指定索引名称并设置索引名称并设置为唯一值特征

[Index("IX_PandaIndex2",IsUnique = true)]
public string ProductType3 { get; set; }

设置多重索引并指定索引的顺序

[Index("IX_PandaIndx1",1,IsUnique = true)]
public string ProductType { get; set; }
[Index("IX_PandaIndex2",2,IsUnique = true)]
public string ProductType2 { get; set; }

设置聚集索引

[Index("IX_ItemObjectUnique", 1, IsClustered = false, IsUnique = true)] 
public int ItemId {get;set;}

反向关联

用于修饰一对多情况下,多方的属性

[InverseProperty("一对多情况下的一方的属性名")]

实例:

//博客Entity
public class Blog
{
    public int BlogId { get; set; }
    public string Name { get; set; }
    //不好的评论,一对多
    public virtual List<Post> BadPost { get; set; }
    //好的评论,一对多
    public virtual List<Post> GoodPosts { get; set; }
}

//评论Entity
public class Post
{
    public int PostId { get; set; }
    public string Title { get; set; }
    public string Content { get; set; }

    public string Url { get; set; }
    //多方,多对一,反向引用修饰
    [InverseProperty("BadPost")]
    public Blog Blog { get; set; }
    //多方,多对一,反向引用修饰
    [InverseProperty("GoodPosts")]
    public Blog Blog2 { get; set; }
}

数据注解与ExtJS中数据验证对应关系

image

image

ASP.NET Core中控制器可以检测数据是否有效

public ViewResult GuestWriteMessage(GusetMessage gusetMessage)
{
    if(ModelState.IsValid)
    {
        return View("FillSuccess", gusetMessage);
    }
    else
    {
        return View();
    }
}

自定义服务器端验证

自定义类继承自ValidationAttribute类型即可

测试使用的模型

using System;
using System.Collections.Generic;
using System.Text;
using System.ComponentModel.DataAnnotations;
using ConsoleApp1.Validator;

namespace ConsoleApp1.Models
{
    public class Post
    {
        [Key]
        public int Id { get; set; }
        [Panda]
        public string Title { get; set; }
    }
}

定义自定义验证

using System;
using System.Collections.Generic;
using System.Text;
using System.ComponentModel.DataAnnotations;
using ConsoleApp1.Models;

namespace ConsoleApp1.Validator
{
    public class PandaAttribute : ValidationAttribute
    {
        public override bool IsValid(object value)
        {
            return base.IsValid(value);
        }

        protected override ValidationResult IsValid(object value, ValidationContext validationContext)
        {
            //将验证对象转为指定的Entity类型
            var post = (Post)validationContext.ObjectInstance;
            if(post.Title == "abc")
            {
                //验证成功
                return ValidationResult.Success;
            }
            else
            {
                //验证失败,返回原因
                return new ValidationResult("这是验证失败的说明");
            }
        }
    }
}

FluentAPI

FluentAPI说明

除了使用惯例原则和属性注解外,还可以使用FluentAPI进行实体类的配置设置

Fluent API 使用Fluent API design pattern设计模式,参考:

https://en.wikipedia.org/wiki/Fluent_interface

FluentAPI提供的功能:

模型配置(Model Configuration)

配置模型到数据库的映射、配置默认的架构(Schema)

配置数据库的函数、存储过程

实体配置(Entity Configuration):

配置实体之间的关系以及映射到表的关系

比如:主键、外键、索引

​ 属性配置(Property Configuration)

配置属性到列的映射

比如:列名、数据类型、默认值、可否为null、外键

与数据注解对比

FluentAPI更加灵活和弹性

注意:如果同时使用FluentAPI和属性注解,则优先级如下:

​ Fluent API > 数据注解 > 默认惯例

由此可以可见,如果设置了FluentAPI,则 数据注解 和 默认惯例 均会被覆盖

常用方法

Configurations Fluent API Methods Usage
Model Configurations HasDbFunction() Configures a database function when targeting a relational database.
HasDefaultSchema() Specifies the database schema.
HasAnnotation() Adds or updates data annotation attributes on the entity.
HasSequence() Configures a database sequence when targeting a relational database.
Entity Configuration HasAlternateKey() Configures an alternate key in the EF model for the entity.
HasIndex() Configures an index of the specified properties.
HasKey() Configures the property or list of properties as Primary Key.
HasMany() Configures the Many part of the relationship, where an entity contains the reference collection property of other type for one-to-Many or many-to-many relationships.
HasOne() Configures the One part of the relationship, where an entity contains the reference property of other type for one-to-one or one-to-many relationships.
Ignore() Configures that the class or property should not be mapped to a table or column.
OwnsOne() Configures a relationship where the target entity is owned by this entity. The target entity key value is propagated from the entity it belongs to.
ToTable() Configures the database table that the entity maps to.
Property Configuration HasColumnName() Configures the corresponding column name in the database for the property.
HasColumnType() Configures the data type of the corresponding column in the database for the property.
HasComputedColumnSql() Configures the property to map to computed column in the database when targeting a relational database.
HasDefaultValue() Configures the default value for the column that the property maps to when targeting a relational database.
HasDefaultValueSql() Configures the default value expression for the column that the property maps to when targeting relational database.
HasField() Specifies the backing field to be used with a property.
HasMaxLength() Configures the maximum length of data that can be stored in a property.
IsConcurrencyToken() Configures the property to be used as an optimistic concurrency token.
IsRequired() Configures whether the valid value of the property is required or whether null is a valid value.
IsRowVersion() Configures the property to be used in optimistic concurrency detection.
IsUnicode() Configures the string property which can contain unicode characters or not.
ValueGeneratedNever() Configures a property which cannot have a generated value when an entity is saved.
ValueGeneratedOnAdd() Configures that the property has a generated value when saving a new entity.
ValueGeneratedOnAddOrUpdate() Configures that the property has a generated value when saving new or existing entity.
ValueGeneratedOnUpdate() Configures that a property has a generated value when saving an existing entity.

使用方式

OnModelCreating方法中使用FluentAPI

通常在DbContext里中的OnModelCreating方法中使用FluentAPI

通过覆写OnModelCreating方法实现配置管理

public class SchoolDBContext: DbContext 
{
    public DbSet<Student> Students { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        //属性配置(Property Configurations)
        modelBuilder.Entity<Student>()
                .Property(s => s.StudentId)
                .HasColumnName("Id")
                .HasDefaultValue(0)
                .IsRequired();
    }
}

链式语句:

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置属性映射到指定的字段名
modelBuilder.Entity<Product>()
.Property(p => p.ProductName)
                     .HasColumnName("Name");
}

Lambda表达式:

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Creator>(entity => {
        entity.Property(e => e.CreatorId)
.HasColumnName("MachineTypeID");
    });
}

定义单独的实体配置类型

除了可以在DbContext对象中的OnModelCreating方法中定义Entity对象的配置。还可以在定义单独的类型定义Entity的配置信息。继承IEntityTypeConfiguration接口即可定义实体配置类,配置实体类和数据库表的对应细节关系。

以Student实体为例,新建StudentEntityConfig类型,继承自IEntityTypeConfiguratior<T>类型。然后再继承的Configure方法中配置实体即可。

using Microsoft.EntityFrameworkCore.Design;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;

namespace PandaTest
{
    public class StudentEntityConfig : IEntityTypeConfiguration<Student>
    {
        public void Configure(EntityTypeBuilder<Student> builder)
        {
            //配置主键
            builder.HasKey(item => item.Id);
            //还可以链式调用
            builder.Property(item => item.Name).HasMaxLength(200);
        }
    }
}

然后在DbContext对象中的OnModelCreating方法中,使用ApplyConfiguration引入单个实体配置对象。

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    //引入单个实体配置
    modelBuilder.ApplyConfiguration(new StudentEntityConfig());
}

或者使用ApplyConfigurationsFromAssembly方法引入所有实体配置对象。

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    //把当前程序集中的所有实体配置都引入
    modelBuilder.ApplyConfigurationsFromAssembly(this.GetType().Assembly);
}

常用配置

设置默认数据库架构

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.HasDefaultSchema("Custom");
}

设置实体映射到指定表

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //配置项
    //设置实体Product映射到表PandaProduct
modelBuilder.Entity<Product>().ToTable("PandaProduct");
}

设置实体映射到指定表并指定架构

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置实体Product映射到表PandaProduct 并指定架构名称
    //注意默认的架构为dbo
modelBuilder.Entity<Product>().ToTable("PandaProduct","Pan");
}

设置表初始数据

modelBuilder.Entity<Person>()
                .HasData(new List<Person>()
                {
                    new Person() { Id = "666", FirstName = "Alabama"}
                });

设置实体映射到表不使用默认约定

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置实体Product不进行默认约定规则配置
modelBuilder.Ignore<Product>();
}

设置属性不进行默认约定规则配置

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置属性不进行默认约定规则配置
modelBuilder.Entity<Product>().Ignore(item => item.ProductType2);
}

设置属性映射到指定的字段名

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置属性映射到指定的字段名
    modelBuilder.Entity<Product>().Property(p => p.ProductName)
        .HasColumnName("Name");
}

或者

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Creator>(entity => {
        entity.Property(e => e.CreatorId).HasColumnName("MachineTypeID");
    });
}

设置属性的映射的字段类型

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置属性的映射的字段类型
    modelBuilder.Entity<Product>().Property(p => p.ProductId)
        .HasColumnType("INT");
}

实例2:

modelBuilder.Entity<OrderDetail>(entity =>
{
    entity.Property(e => e.UnitCost).HasColumnType("money");
});

设置数据类型和默认值

modelBuilder.Entity<Order>(entity =>
{
    entity.Property(e =>
    e.OrderDate).HasColumnType("datetime").HasDefaultValueSql("getdate()");
    entity.Property(e =>
    e.ShipDate).HasColumnType("datetime").HasDefaultValueSql("getdate()");
});

设置默认值为指定值

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Address>().Property(b => b.Country)
                                    .HasDefaultValue("USA");
}

设置默认值为函数或者计算公式

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Person>()
        .Property(b => b.CreatedOn)
        .HasDefaultValueSql("getdate()");
}

设置属性映射为主键

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置属性映射为主键
modelBuilder.Entity<Product>().HasKey(p => p.ProductId);
}

设置主键并指定约束名称

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<LookUp>().HasKey(c => c.Code)
         .HasName("PK_LookUp_Code");
}

设置复合主键

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Person>()
        .HasKey(c => new { c.FirstName, c.LastName });
}

设置主键自增长

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置主键自增长
    modelBuilder.Entity<Product>().Property(p => p.ProductId)
        .HasDatabaseGeneratedOption(System.ComponentModel.DataAnnotations.Schema.DatabaseGeneratedOption.Identity);
}

设置属性为必须

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置属性为必须
    modelBuilder.Entity<Product>().Property(p => p.ProductId)
        .IsRequired();
}

设置属性默认值

使用HasDefaultValue或者HasDefaultValueSql方法

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Person>().
                Property(x => x.CreatedOn)
                .HasDefaultValueSql("getdate()");
}

或者

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Address>()
                .Property(x => x.Country)
                .HasDefaultValue("USA");
}

设置属性的最大长度

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置属性的最大长度
    modelBuilder.Entity<Product>().Property(p => p.ProductName)
        .HasMaxLength(50);
}

设置属性支持Unicode

设置底层使用varchar还是nvarchar

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置属性支持Unicode
    modelBuilder.Entity<Product>().Property(p => p.ProductName)
        .IsUnicode(true);
}

设置字段在数据库中不进行自动生成值

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    //设置字段在数据库中不进行自动生成值
    modelBuilder.Entity<Product>().Property(p => p.ProductName)
        .HasDatabaseGeneratedOption(System.ComponentModel.DataAnnotations.Schema.DatabaseGeneratedOption.None);
}

设置字段索引

modelBuilder.Entity<Customer>(entity =>
{
    entity.HasIndex(e => e.EmailAddress).HasName("IX_Customers").IsUnique();
});

设置计算字段

modelBuilder.Entity<Order>(entity =>
{
entity.Property(e => e.LineItemTotal).HasColumnType("money")
.HasComputedColumnSql("[Quantity]*[UnitCost]");
});

记得在属性上添加自动计算注解

[DatabaseGenerated(DatabaseGeneratedOption.Computed)]

模型级别的查询过滤

modelBuilder.Entity<Order>()
.HasQueryFilter(x => x.CustomerId == CustomerId);

如果需要在使用中Entity不开启查询过滤,可以使用DbSet的IgnoreQueryFilters()方法

配置模型关系并设置外键关系名称

modelBuilder.Entity<SupportLog>(entity => {
    entity.HasOne(d => d.SupportTicket)
        .WithMany(p => p.SupportLog)
        .HasForeignKey(d => d.SupportTicketId)
        .OnDelete(DeleteBehavior.ClientSetNull)
        .HasConstraintName("FK_SupportTicket");
});

设置映射到视图(Fluent API)

modelBuilder.Entity<Blog>().ToView("blogsView");

设置映射的列名称(Fluent API)

builder.Property(item => item.Name).HasColumnName("NameT");

设置映射的列类型(Fluent API)

builder.Property(item => item.Name).HasColumnType("NVARCHAR(200)");

设置映射的列的值自动增加(Fluent API)

//在增加数据时自动增加数据
builder.Property(item => item.Age).ValueGeneratedOnAdd();
//在增加数据时自动增加/更新数据
builder.Property(item => item.Age).ValueGeneratedOnAddOrUpdate();

设置映射的列增加索引(Fluent API)

builder.HasIndex(item => item.Age);

设置映射的列增加索引(复合索引)(Fluent API)

builder.HasIndex(items => new { items.Id, items.Age });

设置映射的列增加索引(唯一索引)(Fluent API)

builder.HasIndex(item => item.Age).IsUnique();

设置映射的列增加索引(聚集索引)(Fluent API)

builder.HasIndex(item => item.Age).IsClustered();

设置映射为Unicode类型(Fluent API)

builder.Property(item=> item.Name).IsUnicode(true);

设置映射默认值(Fluent API)

builder.Property(item => item.Name).HasDefaultValue("Panda");

设置映射必须值(Fluent API)

builder.Property(item => item.Age).IsRequired();

标签:set,get,FlentAPI,Entity,modelBuilder,设置,Data,public
From: https://www.cnblogs.com/cqpanda/p/16783509.html

相关文章