網(wǎng)站首頁 編程語言 正文
一、理解Code First及其約定和配置
傳統(tǒng)設(shè)計(jì)應(yīng)用的方式都是由下而上的,即我們習(xí)慣優(yōu)先考慮數(shù)據(jù)庫,然后使用這個(gè)以數(shù)據(jù)為中心的方法在數(shù)據(jù)之上構(gòu)建應(yīng)用程序。這種方法非常適合于數(shù)據(jù)密集的應(yīng)用或者數(shù)據(jù)庫很可能包含多個(gè)應(yīng)用使用的業(yè)務(wù)邏輯的應(yīng)用。對(duì)于這種應(yīng)用,如果要使用EF的話,我們必須使用Database First方式。
設(shè)計(jì)應(yīng)用的另一種方法就是以領(lǐng)域?yàn)橹行牡姆绞剑I(lǐng)域驅(qū)動(dòng)設(shè)計(jì)DDD)。DDD是一種由上而下的方式,我們通過從實(shí)現(xiàn)應(yīng)用所需要的領(lǐng)域模型和實(shí)體的角度思考,從而開始設(shè)計(jì)應(yīng)用。數(shù)據(jù)庫很少用來用于領(lǐng)域模型數(shù)據(jù)的持久化。使用DDD意味著我們要根據(jù)每個(gè)應(yīng)用的需求來設(shè)計(jì)模型和實(shí)體,而且模型和實(shí)體是數(shù)據(jù)庫可忽略的,即可以使用任何數(shù)據(jù)庫技術(shù)實(shí)現(xiàn)保存。在這些情景中,我們應(yīng)該使用EF的Code First方式,因?yàn)樗试S我們創(chuàng)建POCO(Plain Old CLR Objects)作為持久化可忽略的領(lǐng)域模型。
使用Code First的優(yōu)勢(shì)在于:
- 1、支持DDD。
- 2、可以早早地著手開發(fā),因?yàn)槲覀儾槐氐却龜?shù)據(jù)庫的創(chuàng)建。
- 3、持久化層(底層的數(shù)據(jù)庫)的改變不會(huì)對(duì)現(xiàn)有的模型有任何影響。
我們需要搞清楚的第一件事就是約定大于配置的概念。Code First方式期望模型類遵守一些約定,這樣的話數(shù)據(jù)庫持久化邏輯就可以從模型中提取出來。比如,如果我們給一個(gè)模型定義了一個(gè)Id屬性,那么它就會(huì)映射到數(shù)據(jù)庫中該類所對(duì)應(yīng)的那張表的主鍵。這種基于約定的方式的好處在于:如果我們遵守了這些約定,那么我們就不必寫額外的代碼來管理數(shù)據(jù)庫持久邏輯。但這樣也存在缺點(diǎn),缺點(diǎn)在于:如果沒有遵守某個(gè)約定,那么EF就不會(huì)從模型中提取到需要的信息,運(yùn)行時(shí)會(huì)拋異常。
在這種沒有遵守約定又要持久化數(shù)據(jù)的情況下,我們需要使用Code First的配置項(xiàng)提供關(guān)于模型一些額外的信息。比如,如果我們的模型類中沒有Id屬性作為主鍵,那么我們需要在想要的屬性上加上[Key]特性,這樣它就會(huì)被當(dāng)作主鍵了。
注意:
EF使用模型類的復(fù)數(shù)的約定來創(chuàng)建數(shù)據(jù)表名,創(chuàng)建的列名和該類的屬性名是一樣的。
示例:
使用在EF應(yīng)用一:Code First模式中介紹的使用方法創(chuàng)建一個(gè)數(shù)據(jù)庫,其中實(shí)體類Product定義如下:
using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace EFAppointmentCreateTable.Model { public class Product { public int Id { get; set; } public string ProductName { get; set; } public double ProductPrice { get; set; } } }
數(shù)據(jù)庫上下文Context類定義如下:
using EFAppointmentCreateTable.Model; using System; using System.Collections.Generic; using System.Data.Entity; using System.Linq; using System.Text; namespace EFAppointmentCreateTable.EFContext { ////// 定義數(shù)據(jù)庫上下文類,該類繼承自DbContext /// public class Context:DbContext { ////// 定義構(gòu)造函數(shù),繼承自父類的構(gòu)造函數(shù),通過繼承父類的構(gòu)造函數(shù)來創(chuàng)建數(shù)據(jù)庫 /// 父類構(gòu)造函數(shù)的參數(shù)是配置文件中配置的數(shù)據(jù)庫連接字符串 /// public Context() : base("DbConnection") { } //實(shí)體屬性集合 根據(jù)實(shí)體名稱的復(fù)數(shù)形式生成數(shù)據(jù)庫的表名 // 生成的表名是Products 和屬性名Products無關(guān) // 即使把屬性名改成Product,生成的表名還是Products 約定大于配置 public DbSetProducts { get; set; } } }
創(chuàng)建后的數(shù)據(jù)庫截圖如下:
從上面的截圖中可以看出,生成的數(shù)據(jù)庫表名是實(shí)體Product的復(fù)數(shù)形式,使用默認(rèn)約定生成Id主鍵列。
二、使用數(shù)據(jù)注解創(chuàng)建表結(jié)構(gòu)
1、.NET數(shù)據(jù)類型和SQL數(shù)據(jù)類型之間的映射
首先EF這個(gè)ORM工具就是用來解決.NET類型和SQL Server列類型之間的阻抗失配的問題。比如,假設(shè)你在.net中定義了一個(gè)int類型的屬性,那么你就可以認(rèn)為EF已經(jīng)安全地處理了這個(gè)列的定義并使用了合適的類型與之對(duì)應(yīng)。記住一些.NET類型和SQL Server列類型之間的映射是很有必要的,下面是一些最常用的映射關(guān)系:
SQL Server Database Type | .NET Type |
---|---|
Bigint | Int64 |
binary,varbinary | Byte[] |
Bit | Boolean |
date,datetime, datetime2,smalldatetime |
DateTime |
Datetimeoffset | DateTimeOffset |
decimal,money smallmoney,numeric |
Decimal |
float | Double |
int | Int32 |
nchar nvarchar,char varchar |
String |
real | Single |
rowversion,timestamp | Byte[] |
smallint | Int16 |
time | TimeSpan |
tinyint | Byte |
uniqueidentifier | Guid |
完整的映射列表可以參考SQL Server數(shù)據(jù)類型映射,如果你使用的是其他類型的數(shù)據(jù)庫,你可以在網(wǎng)上自行查找。比如,,如果是Oracle,那么你可以在這里查找:oracle數(shù)據(jù)類型映射。
2、配置原始屬性
以.NET中的string類型的屬性開始討論。SQL Server中的很多類型都會(huì)映射到.NET中的string類型,其他主流的RDBMS也是一樣的。因此,決定如何存儲(chǔ)字符串類型的信息是很重要的,很多關(guān)系數(shù)據(jù)庫管理引擎都有多個(gè)字符存儲(chǔ)類型,他們通常都有以字母N打頭的字符類型,這個(gè)字母表示要存在這些列中的數(shù)據(jù)是Unicode數(shù)據(jù),基于每個(gè)字符以2個(gè)字節(jié)的格式存儲(chǔ)。因此,如果你的數(shù)據(jù)庫中的列存儲(chǔ)的是英文的話,就可以使用varchar或者char(可能會(huì)加速查詢),如果使用的是中文的話,就要使用nvarchar或者nchar。此外,還可以使用帶有var的字符類型來指定列的長度是可變的,不使用var的話,字符長度是不可變的。
在EF中有以下幾種配置數(shù)據(jù)庫結(jié)構(gòu)的方式,分別是:
- 1、特性,也叫數(shù)據(jù)注解。
- 2、DbModelBuilder API。
- 3、配置伙伴類。
3、特性(數(shù)據(jù)注解)
這些特性類都是.NET的一部分,位于System.ComponentModel.DataAnnotaions命名空間下。下面我們修改在上一個(gè)例子中使用的Product實(shí)體類,修改后的Product實(shí)體類如下:
using System; using System.Collections.Generic; using System.ComponentModel.DataAnnotations; using System.ComponentModel.DataAnnotations.Schema; using System.Linq; using System.Text; using System.Threading.Tasks; namespace EFDataAnnotations.Model { ////// 使用Table特性指定生成的表名 /// [Table("Product")] public class Product { //使用Key特性指示該列是主鍵列 [Key] //使用Column特性指示生成的數(shù)據(jù)庫表中的列名 [Column("ID")] public int ProductId { get; set; } ////// StringLength設(shè)置數(shù)據(jù)庫表中列的長度 /// 指示ProductName字段的最大長度是10,最小長度是2 /// [StringLength(10,MinimumLength=2)] public string ProductName { get; set; } public double ProductPrice { get; set; } ////// DataType特性指示生成的數(shù)據(jù)庫表中列的數(shù)據(jù)類型 /// [DataType(DataType.DateTime)] public DateTime ProductionTime { get; set; } ////// NotMapped特性指示不把該屬性映射到數(shù)據(jù)庫表中,即在生成的表中沒有d這一列 /// [NotMapped] public decimal d { get; set; } } }
運(yùn)行程序后生成的數(shù)據(jù)庫:
修改代碼之后,我們將表的名字使用Table特性全部重新命名成了但是,將Product的主鍵通過Column特性更改為Id,Key特性指定它是主鍵,還通過StringLength指定了ProductName列的最大長度為10個(gè)字符,最小為2個(gè)字符。
此外,數(shù)據(jù)注解也可以用作驗(yàn)證特性。如果持久化數(shù)據(jù)時(shí),模型對(duì)象的屬性值和數(shù)據(jù)注解所標(biāo)記的不一致,就會(huì)拋異常。
數(shù)據(jù)驗(yàn)證相關(guān)的數(shù)據(jù)注解:
特性 | 解釋 |
---|---|
Remote | 使用 jQuery 驗(yàn)證插件遠(yuǎn)程驗(yàn)證程序的特性 |
FileExtension | 驗(yàn)證文件擴(kuò)展名 |
Compare | 比較兩個(gè)屬性的值 |
RegularExpression | 使用正則表達(dá)式驗(yàn)證 |
CustomValidation | 自定義驗(yàn)證方法 |
DataType | 指定要與數(shù)據(jù)字段關(guān)聯(lián)的附加類型的名稱 |
EmailAddress | 電子郵件地址(相當(dāng)于DataType(DataType.Email)) |
Phone | 電話(相當(dāng)于DataType(DataType.Phone)) |
CreditCard | 信用卡號(hào)碼(相當(dāng)于DataType(DataType.CreditCard)) |
Url | 驗(yàn)證URL(相當(dāng)于DataType(DataType.Url)) |
MemberShipPassword | 驗(yàn)證密碼字段是否滿足成員資格提供程序的當(dāng)前密碼要求 |
數(shù)據(jù)映射相關(guān)的數(shù)據(jù)注解:
特性 | 解釋 |
---|---|
Key | 主鍵字段 |
Column | 數(shù)據(jù)庫列屬性映射 |
NotMapped | 不要?jiǎng)?chuàng)建對(duì)應(yīng)的字段 |
Table | 指定類將映射到的數(shù)據(jù)庫表 |
ForeignKey | 表示關(guān)系中用作外鍵的屬性 |
DatabaseGenerated |
指定屬性應(yīng)該映射到數(shù)據(jù)表中計(jì)算的列。也可以用于映射到自動(dòng)增長的數(shù)據(jù)庫表。 指定數(shù)據(jù)庫生成屬性值的方式(EF不追蹤屬性的變化) |
Required | 必填字段 |
MaxLength | 指定屬性中允許的數(shù)組或字符串?dāng)?shù)據(jù)的最大長度 |
MinLength | 指定屬性中允許的數(shù)組或字符串?dāng)?shù)據(jù)的最小長度 |
StringLength | 指定最小和最大字符長度 |
Range | 指定數(shù)值范圍? |
數(shù)據(jù)顯示相關(guān)的數(shù)據(jù)注解:?
特性 | 解釋 |
---|---|
DisplayName | 指定本地化的字符串(習(xí)慣用語類) |
Display | 指定本地化的字符串(習(xí)慣用語屬性) |
DisplayFormat | 設(shè)置數(shù)據(jù)字段的格式 |
ReadOnly | 指定該特性所綁定到的屬性是只讀屬性還是讀/寫屬性 |
EditAble | 指示數(shù)據(jù)字段是否可編輯 |
HiddenInput | 指示是否應(yīng)將屬性值或字段值呈現(xiàn)為隱藏的 input 元素 |
ScaffoldColumn | 指定類或數(shù)據(jù)列是否使用基架 |
UIHint | 指定動(dòng)態(tài)數(shù)據(jù)用來顯示數(shù)據(jù)字段的模板 |
其他:
特性 | 解釋 |
---|---|
DisplayColumn | 將所引用的表中顯示的列指定為外鍵列 |
Description | 可視化設(shè)計(jì)器在引用組件成員時(shí)可以顯示指定的說明 (命名空間:System.ComponentModel.DescriptionAttribute) |
原文鏈接:https://www.cnblogs.com/dotnet261010/p/7417853.html
相關(guān)推薦
- 2023-10-14 c/c++--編譯指令(預(yù)處理之后) #pragma
- 2022-07-17 代碼解析python標(biāo)準(zhǔn)庫logging模塊_python
- 2022-04-11 shell腳本根據(jù)進(jìn)程查找指定容器的方法_linux shell
- 2023-07-04 springboot引入外部sdk,以及在maven中配置,以及連同sdk打包
- 2024-03-15 Spring Framework對(duì)DAO(Data Access Object)的支持
- 2023-05-22 shell腳本自動(dòng)輸入用戶名和密碼的實(shí)現(xiàn)_linux shell
- 2022-10-07 Flutter?GetPageRoute實(shí)現(xiàn)嵌套導(dǎo)航學(xué)習(xí)_IOS
- 2022-12-13 iOS底層實(shí)例解析Swift閉包及OC閉包_IOS
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支