(20)ASP.NET Core2.2 EF创建模型(必需属性和可选属性、最大长度、并发标记、阴影属性) - 指南

news/2025/10/14 13:46:02/文章来源:https://www.cnblogs.com/ljbguanli/p/19140925

(20)ASP.NET Core2.2 EF创建模型(必需属性和可选属性、最大长度、并发标记、阴影属性) - 指南

1.必需和可选属性

如果实体属性可以包含null,则将其视为可选。如果属性的有效值不可以包含null,则将其视为必需属性。映射到关系数据库架构时,必需的属性将创建为不可为null的列,而可选属性则创建为可以为null的列。

1.1约定

按照约定,.NET 类型可以包含null的属性将配置为可选,而.NET类型不包含null的属性将根据需要进行配置。例如,具有.net值类型(int、decimal、bool等)的所有属性都是必需的,而具有可为null的.net值类型(int?、decimal?、bool?等)的所有属性都是配置为可选。

1.2数据批注

可以按如下所示将"约定"可以为"可选"的属性配置为"必需":

namespace EFModeling.DataAnnotations.Required
{
    class MyContext : DbContext
    {
        public DbSet Blogs { get; set; }
    }
    public class Blog
    {
        public int BlogId { get; set; }
        //加上这个批注,这个值就必需写入
        [Required]
        public string Url { get; set; }
    }
}

1.3Fluent API

namespace EFModeling.FluentAPI.Required
{
    class MyContext : DbContext
    {
        public DbSet Blogs { get; set; }
        protected override void OnModelCreating(ModelBuilder modelBuilder)
        {
            modelBuilder.Entity()
                .Property(b => b.Url)
                //这个方法表示必需写入
                .IsRequired();
        }
    }
    public class Blog
    {
        public int BlogId { get; set; }
        public string Url { get; set; }
    }
}

2.最大长度

配置最大长度可为数据存储提供有关要对给定属性使用的相应数据类型的提示。最大长度仅适用于数组数据类型,如string和byte[]。例如前端传统数据长度远大于限定的长度,则提示。

2.1约定

按照约定,应由数据库提供程序为属性选择适当的数据类型,即数据库字段设置长度多少,生产程序实体接受值时就限定长度多少。对于具有长度的属性,数据库提供程序通常将选择允许最长数据长度的数据类型。例如,Microsoft SQL Server将对字符string属性使用 nvarchar(max)(如果该列用作键,则会使用nvarchar(450))。

2.2数据批注

你可以使用数据批注为属性配置最大长度。此示例面向SQL Server,因此使用数据类型 nvarchar(500):

namespace EFModeling.DataAnnotations.MaxLength
{
    class MyContext : DbContext
    {
        public DbSet Blogs { get; set; }
    }
    public class Blog
    {
        public int BlogId { get; set; }
        //设置最大长度
        [MaxLength(500)]
        public string Url { get; set; }
    }
}

2.3Fluent API

namespace EFModeling.FluentAPI.MaxLength
{
    class MyContext : DbContext
    {
        public DbSet Blogs { get; set; }
        protected override void OnModelCreating(ModelBuilder modelBuilder)
        {
            modelBuilder.Entity()
                .Property(b => b.Url)
                //设置最大长度
                .HasMaxLength(500);
        }
    }
    public class Blog
    {
        public int BlogId { get; set; }
        public string Url { get; set; }
    }
}

3.并发标记

当我们发现生产环境某个实体字段经常处于并发当中,我们可以批注一下为并发字段。

3.1约定

按照约定,属性永远不会配置为并发标记。

3.2数据注释

您可以使用数据批注将属性配置为并发标记:

public class Person
{public int PersonId { get; set; }//并发标记
    [ConcurrencyCheck]
    public string LastName { get; set; }
    public string FirstName { get; set; }
}

3.3Fluent API

您可以使用熟知的API将属性配置为并发标记:

class MyContext : DbContext
{
    public DbSet People { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity()
            .Property(p => p.LastName)
            //并发标记
            .IsConcurrencyToken();
    }
}
public class Person
{
    public int PersonId { get; set; }
    public string LastName { get; set; }
    public string FirstName { get; set; }
}

4.时间戳/行版本

时间戳是一个属性类型,在每次插入或更新行时,数据库都会生成一个新值。此该属性类型也被视为并发标记。这可以确保在你和其他人修改了行数据时你会收到异常信息。

4.1约定

按照约定,属性永远不会配置为时间戳。

4.2数据注释

你可以使用数据批注将属性配置为时间戳:

public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }
    //设置时间戳
    [Timestamp]
    public byte[] Timestamp { get; set; }
}

4.3Fluent API

你可以使用熟知的API将属性配置为时间戳:

class MyContext : DbContext
{
    public DbSet Blogs { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity()
            .Property(p => p.Timestamp)
            //设置时间戳
            .IsRowVersion();
    }
}
public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }
    public byte[] Timestamp { get; set; }
}

5.阴影属性

当数据库中的数据不应在映射的实体类型上公开时,阴影属性非常有用。它们最常用于外键属性,其中两个实体之间的关系由数据库中的外键值表示,但使用实体类型之间的导航属性在实体类型上管理关系,可以通过ChangeTracker API获取和更改影子属性值:

context.Entry(myBlog).Property("LastUpdated").CurrentValue = DateTime.Now;

可以通过EF.Property静态方法在LINQ查询中引用影子属性:

var blogs = context.Blogs.OrderBy(b => EF.Property(b, "LastUpdated"));

5.1约定

如果发现了关系,但在依赖实体类中找不到外键属性,则可以按约定创建阴影属性。在这种情况下,将引入阴影外键属性。影子外键属性将命名<navigation property name><principal key property name>为(指向主体实体的依赖实体上的导航用于命名)。如果主体键属性名称包含导航属性的名称,则该名称将只是<principal key property name>。如果依赖实体上没有导航属性,则会在其位置使用主体类型名称。例如,下面的代码列表将导致BlogId Post向实体引入阴影属性:

class MyContext : DbContext
{
    public DbSet Blogs { get; set; }
    public DbSet Posts { get; set; }
}
public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }
    //阴影属性
    public List Posts { get; set; }
}
public class Post
{
    public int PostId { get; set; }
    public string Title { get; set; }
    public string Content { get; set; }
     //阴影属性
    public Blog Blog { get; set; }
}

5.2数据注释

不能通过数据批注创建阴影属性。

5.3Fluent API

你可以使用"熟知API"配置阴影属性。一旦你调用了Property方法的字符串重载,就可以链接到其他属性的任何配置调用。如果提供Property方法的名称与现有属性的名称相匹配(一个阴影属性或在实体类中定义的属性),则代码将配置该现有属性,而不是引入新的阴影属性:

class MyContext : DbContext
{
    public DbSet Blogs { get; set; }
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity()
            //创建阴影属性
            .Property("LastUpdated");
    }
}
public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }
}

参考文献:
必需和可选属性https://learn.microsoft.com/zh-cn/ef/core/modeling/entity-properties?tabs=data-annotations%2Cwithout-nrt#required-and-optional-properties
最大长度https://learn.microsoft.com/zh-cn/ef/core/modeling/entity-properties?tabs=data-annotations%2Cwith-nrt#maximum-length
并发标记https://learn.microsoft.com/zh-cn/ef/core/saving/concurrency?tabs=data-annotations
阴影属性https://learn.microsoft.com/zh-cn/ef/core/modeling/shadow-properties

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/936799.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

(在构造函数中)调用super(props)的目的是什么?

在 React 类组件 中,constructor 构造函数里调用 super(props) 是一个非常经典的机制, 一、基本背景 在 React 中,我们定义类组件时通常这样写: class MyComponent extends React.Component {constructor(props) {…

温故知新,机器人进化论,机器人分类与全球格局

机器人基本概念与范畴 “机器人”是一个比较宽泛的概念,不同文献和标准下的定义稍有差异。一般来说,机器人是能感知环境、作出决策或规划、并作用于环境的机械系统(带有驱动、传感、控制、算法等)的一类设备。 在工…

Zemax:初学者的混合模式 - 指南

Zemax:初学者的混合模式 - 指南pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco&qu…

【无标题】使用 Playwright 实现跨 Chromium、Firefox、WebKit 浏览器自动化管理

【无标题】使用 Playwright 实现跨 Chromium、Firefox、WebKit 浏览器自动化管理pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-…

西门子博图软件TIA V18使用PLCSIM Advanced V5.0进行仿真与其他程序进行通讯

由于标准版PLCSIM只是PLC内部通讯用,若要与外部进行通讯,需要安装Advanced版本,TIA V18的兼容advanced版本需要在V5.0以上,这里使用了V5.0版本。 PLCSIM Advanced V5.0安装过程中可能会有兼容性之类的报错 发生该…

MyEclipse 2017/2018 安装与破解 图文教程

SSM 框架-02-MyEclipse 2017/2018 安装与破解 现在在学J2EE,然后使用的工具就是 MyEclipse,现在就抛弃 Eclipse 了,我就不多说它俩的区别了,但是 MyEclipse 是收费的,下面介绍 MyEclipse 2017 安装与破解 一、下载…

面向对象初级

面向对象初级:1.面向对象三大特征:封装  继承   多态    self含义:  指向实例对象本身,让实例能够访问类中的属性和方法2.类和对象的创建:    类和对象的创建(登录注册功能) class User:def __in…

【文章目录】

文章目录 【STM32】 【STM32系列】STM32通用【STM32系列】基于HAL库的串口DMA空闲中断接收+串口DMA发送 【STM32系列】超好用的开源按键状态系统lwbtn,以及超详细的移植教程 【STM32系列】EXTI11触发三ADC同步(并非同…

Excel DDE 教學:即時資料交換的詳細指南 - 指南

Excel DDE 教學:即時資料交換的詳細指南 - 指南pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", …

子网掩码基础知识

子网掩码基础知识 子网掩码由连续的1和连续的0组成,1表示网络位,0表示主机位。 1. 基础概念IPv4地址:由32位二进制组成,通常表示为4个十进制数(如192.168.1.0)。 子网掩码:用于划分网络和主机部分。例如,255.2…

iOS 框架全解析,原生框架与跨平台框架对比、开发应用打包与 App Store 上架实战经验 - 指南

iOS 框架全解析,原生框架与跨平台框架对比、开发应用打包与 App Store 上架实战经验 - 指南pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !impor…

微信机器人框架

微信机器人框架、微信二次开发机器人接口开发、微信机器人制作教程API文档 接入指南、对接流程 1.申请api平台账号 2.开通接口权限 3.对接api 4.测试上线如何测试? 假如需要接收并处理微信消息,研发人员需掌握任意一…

AI元人文构想基础理论体系研究

AI元人文构想基础理论体系研究 ——2025.10.14修订版引言:AI元人文构想的理论溯源与概念界定1.1 理论背景与提出缘由 当前人工智能发展正面临根本性挑战:主流AI范式在有效处理人类价值的复杂性、模糊性和动态性方面存…

详细介绍:Go 语言中指针介绍

详细介绍:Go 语言中指针介绍pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco"…

实用指南:JavaWeb 课堂笔记 —— 24 AOP 面向切面编程

实用指南:JavaWeb 课堂笔记 —— 24 AOP 面向切面编程pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas&q…

2025年7款与Jira数据同步的实用国产优秀项目管理软件对比

为什么我们需要这篇对比? 如果你是一位项目经理,或许经历过这些困扰:团队用惯了Jira,但因数据合规要求急需国产替代,却担心同步成本高、功能断层? 想找一款能与Jira无缝对接的工具,兼顾本地化服务与灵活扩展,却…

ESP8266 PMW使用的简单介绍

前言 呼吸灯是常见的LED应用。LED呼吸灯的流程是缓缓点亮LED灯,再缓缓熄灭LED灯。在夜色下,缓慢闪烁的LED显得格外迷人。 如何使用ESP8266MOD实现此效果, 这里简单地总结一下。 一、PWM单元介绍ESP8266有四个PWM输出…

DevEco Testing全面解析:HarmonyOS测试框架与实战指南 - 教程

DevEco Testing全面解析:HarmonyOS测试框架与实战指南 - 教程pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Con…

C#知识学习-015(修饰符_4) - 详解

C#知识学习-015(修饰符_4) - 详解pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco…