更新记录
转载请注明出处:
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;
所有注解:
注意:
数据注解并不属于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中数据验证对应关系
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