DataAnnotations
用於配置模型類,它將突出顯示最常用的配置。 DataAnnotations
也被許多.NET應用程式所理解,例如ASP.NET MVC,它允許這些應用程式利用相同的注釋來進行客戶端驗證。DataAnnotation
屬性重寫默認的Code-First
約定。
System.ComponentModel.DataAnnotations
包括以下影響列的可空性或大小的屬性。
- Key
- Timestamp
- ConcurrencyCheck
- Required
- MinLength
- MaxLength
- StringLength
System.ComponentModel.DataAnnotations.Schema
命名空間包括以下影響資料庫模式的屬性。
- Table
- Column
- Index
- ForeignKey
- NotMapped
- InverseProperty
1. 鍵(Key)
實體框架(Entity Framework或簡稱為EF )依賴於具有用於跟蹤實體的鍵值的每個實體。 Code First依賴的其中一個約定是它如何暗示哪個屬性是每個Code First類中的鍵。
約定是尋找一個名為Id
的屬性,或者將類名和Id
結合起來的屬性,比如StudentId
。 該屬性將映射到資料庫中的主鍵列。學生,課程和入學課程遵循這個約定。
現在讓假設Student
類使用名稱StdntID
而不是ID
。 當Code First找不到符合此約定的屬性時,它將拋出一個異常,因為Entity Framework要求必須具有一個鍵屬性。
可以使用鍵注釋來指定哪個屬性將被用作EntityKey
。
下麵來看看包含StdntID
的Student
類。 它不遵循默認的Code First約定,所以要處理這個問題,添加了Key
屬性,使StdntID
列成為主鍵。
public class Student{
[Key]
public int StdntID { get; set; }
public string LastName { get; set; }
public string FirstMidName { get; set; }
public DateTime EnrollmentDate { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
在運行應用程式並查看SQL Server資源管理器中的資料庫時,您將看到現在Students
表中的主鍵是:StdntID
。
實體框架(Entity Framework)也支持複合鍵。 複合鍵是由多個屬性組成的主鍵。例如,有一個DrivingLicense
類,其主鍵是LicenseNumber
和IssuingCountry
的組合。
public class DrivingLicense{
[Key, Column(Order = 1)]
public int LicenseNumber { get; set; }
[Key, Column(Order = 2)]
public string IssuingCountry { get; set; }
public DateTime Issued { get; set; }
public DateTime Expires { get; set; }
}
當有組合鍵時,實體框架要求你定義鍵屬性的順序。可以使用Column
注釋來指定順序。
2. 時間戳(Timestamp)
Code First會將Timestamp
屬性視為ConcurrencyCheck
屬性,但它也將確保Code First生成的資料庫字段不可空。
使用rowversion
或timestamp
字段進行併發檢查更為常見。但是,不要使用ConcurrencyCheck
注釋,只要屬性的類型是位元組數組,就可以使用更具體的TimeStamp
注釋。在給定的類中只能有一個時間戳屬性。
下麵來看一個簡單的例子,將TimeStamp
屬性添加到Course
類中。參考以下代碼 -
public class Course{
public int CourseID { get; set; }
public string Title { get; set; }
public int Credits { get; set; }
[Timestamp]
public byte[] TStamp { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
如上例所示,Timestamp
屬性應用於Course
類的Byte []
屬性。 所以,Code First 將在Courses
表中創建一個時間戳列TStamp
。
3. ConcurrencyCheck
ConcurrencyCheck
注釋允許在用戶編輯或刪除實體時標記一個或多個要用於資料庫併發檢查的屬性。如果一直在使用EF Designer,那麼這將與將屬性的“併發性模式”設置為“固定”一致。
下麵來看看一個簡單的例子,通過將它添加Title
屬性到Course
類中來瞭解ConcurrencyCheck
是如何工作的。
public class Course{
public int CourseID { get; set; }
[ConcurrencyCheck]
public string Title { get; set; }
public int Credits { get; set; }
[Timestamp, DataType("timestamp")]
public byte[] TimeStamp { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
在上面的Course
類中,ConcurrencyCheck
屬性應用於現有的Title
屬性。 Code First將在update
命令中包含Title
列來檢查以下代碼所示的樂觀併發。
exec sp_executesql N'UPDATE [dbo].[Courses]
SET [Title] = @0
WHERE (([CourseID] = @1) AND ([Title] = @2))
',N'@0 nvarchar(max) ,@1 int,@2 nvarchar(max)
',@0 = N'Maths',@1 = 1,@2 = N'Calculus'
go
4. Required
Required
注釋告訴實體框架(Entity Framework)需要一個特定的屬性。下麵來看看Student
類,其中必需的id
被添加到FirstMidName
屬性中。 Required
屬性將強制Entity Framework 確保該屬性中包含數據。
public class Student{
[Key]
public int StdntID { get; set; }
[Required]
public string LastName { get; set; }
[Required]
public string FirstMidName { get; set; }
public DateTime EnrollmentDate { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
可以在上面的Student
類的示例中看到Required
屬性應用於FirstMidName
和LastName
。 因此,Code First將在Students
表中創建一個NOT NULL
的 FirstMidName
和LastName
列,如下圖所示。
5. MaxLength
MaxLength
屬性用於指定其他屬性驗證。它可以應用於類的字串或數組類型的屬性。 Entity Framework的 Code First 將設置MaxLength
屬性中指定的列的大小。
下麵來看看MaxLength(24)
屬性應用於Title
屬性的以下Course
類。
public class Course{
public int CourseID { get; set; }
[ConcurrencyCheck]
[MaxLength(24)]
public string Title { get; set; }
public int Credits { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
當運行上述應用程式時,Code-First將在Coursed
表中創建一個nvarchar(24)
列標題,如以下螢幕截圖所示。
現在當用戶設置包含超過24
個字元的標題時,Entity Framework將拋出EntityValidationError
異常。
6. MinLength
MinLength
屬性可指定其他屬性驗證,就像上面使用的MaxLength
屬性一樣。 MinLength
屬性也可以與MaxLength
屬性一起使用,如下面的代碼所示。
public class Course{
public int CourseID { get; set; }
[ConcurrencyCheck]
[MaxLength(24) , MinLength(5)]
public string Title { get; set; }
public int Credits { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
如果在MinLength
屬性中將Title
屬性的值設置為小於指定的長度或大於MaxLength
屬性中的指定長度,則EF將拋出EntityValidationError
異常。
7. StringLength
StringLength
還允許指定其他屬性驗證,如MaxLength
。 不同的是StringLength
屬性只能應用於Domain
類的字串類型屬性。參考以下示例代碼 -
public class Course{
public int CourseID { get; set; }
[StringLength (24)]
public string Title { get; set; }
public int Credits { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
Entity Framework還驗證StringLength
屬性的屬性值。 現在,如果用戶設置標題(Title),其中包含超過24
個字元,那麼EF將拋出EntityValidationError
異常。
8. Table
默認代碼第一個約定創建一個與類名相同的表名。 如果讓Code First創建資料庫,則還可以更改正在創建的表的名稱。可以讓Code First使用現有的資料庫表。 但並不總是這樣,有時類的名稱與資料庫中表的名稱不能總是相匹配。
Table
屬性重寫此默認約定。 對於給定的域類,EF Code First將在Table
屬性使用指定的名稱來創建一個表。
下麵來看看一個類名為Student
的例子,按照慣例,Code First假定這將映射到一個名為Students
的表。 如果不是這種情況,可以使用Table
屬性指定表的名稱,如以下代碼所示 指定要創建的表名稱為:StudentInfo -
[Table("StudentsInfo")]
public class Student{
[Key]
public int StdntID { get; set; }
[Required]
public string LastName { get; set; }
[Required]
public string FirstMidName { get; set; }
public DateTime EnrollmentDate { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
現在可以看到Table
屬性將表指定為StudentsInfo
。 生成表時,如下圖所示的表名StudentInfo
。
不僅可以指定表名,還可以使用以下代碼使用Table
屬性指定表的模式。
[Table("StudentsInfo", Schema = "Admin")]
public class Student{
[Key]
public int StdntID { get; set; }
[Required]
public string LastName { get; set; }
[Required]
public string FirstMidName { get; set; }
public DateTime EnrollmentDate { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
在上面的例子中,表被指定為Admin模式。 現在,Code First將在Admin模式中創建StudentsInfo
表,如以下螢幕截圖所示。
9. Column
Column
屬性也與Table
屬性相同,但Table
屬性覆蓋表行為,而Column
屬性覆蓋列行為。 默認代碼第一個約定創建一個與屬性名相同的列名。
如果讓Code First創建資料庫,並且還希望更改表中列的名稱。Column
屬性重寫此默認約定。 EF Code First將在給定類屬性的Column
屬性中創建一個具有指定名稱的列。
下麵來看看下麵的例子,其中屬性名為FirstMidName
,按照慣例,Code First假定這將映射到一個名為FirstMidName
的列。 如果不是要映射到FirstMidName
列時,可以使用Column
屬性指定其他列的名稱,如以下代碼所示。
public class Student{
public int ID { get; set; }
public string LastName { get; set; }
[Column("FirstName")]
public string FirstMidName { get; set; }
public DateTime EnrollmentDate { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
現在可以看到Column
屬性將列指定為FirstName
。 生成表後,可以看到列名為FirstName
,如以下螢幕截圖所示。
10. Index
Index
屬性是在Entity Framework 6.1中引入的。
注 - 如果您使用的是早期版本,則本節中的資訊不適用。
可以使用Index
屬性在一列或多列上創建索引。將屬性添加到一個或多個屬性將導致EF在創建資料庫時在資料庫中創建相應的索引。
在大多數情況下,索引使數據的檢索更快,更高效。但是,使用索引重載表或視圖可能會不愉快地影響其他操作(如插入或更新)的性能。
索引是實體框架中的新功能,可以通過減少從資料庫查詢數據所需的時間來提高Code First應用程式的性能。
可以使用Index
屬性將索引添加到資料庫,並覆蓋默認的“唯一”和“群集”設置,以獲得最適合您的方案的索引。默認情況下,索引將被命名為IX_<屬性名稱>
讓我們來看看以下代碼,其中Index
屬性被添加到Course
類Credits
列上。
public class Cours{
public int CourseID { get; set; }
public string Title { get; set; }
[Index]
public int Credits { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
可以看到Index
屬性應用於Credits
屬性。 現在,當表生成時,將在索引中看到名稱為IX_Credits
的索引。
默認情況下,索引是非唯一的,但是可以使用IsUnique
命名參數來指定索引應該是唯一的。 以下示例引入了一個唯一索引,如下面的代碼所示。
public class Course{
public int CourseID { get; set; }
[Index(IsUnique = true)]
public string Title { get; set; }
[Index]
public int Credits { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
11. ForeignKey
Code First約定將處理模型中最常見的關係。 例如,通過更改Student
類中的key
屬性名稱,創建了與Enrollment
類的關係問題。
public class Enrollment{
public int EnrollmentID { get; set; }
public int CourseID { get; set; }
public int StudentID { get; set; }
public Grade? Grade { get; set; }
public virtual Course Course { get; set; }
public virtual Student Student { get; set; }
}
public class Student{
[Key]
public int StdntID { get; set; }
public string LastName { get; set; }
public string FirstMidName { get; set; }
public DateTime EnrollmentDate { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
在生成資料庫時,Code First會在Enrollment
類中看到StudentID
屬性,並按照約定將其識別為類名稱加ID
,作為Student
類的外鍵。但是Student
類中沒有StudentID
屬性,而是Student
類中的StdntID
屬性。
解決方法是在Enrollment
中創建導航屬性,並使用ForeignKey DataAnnotation
來幫助Code First瞭解如何構建兩個類之間的關係,如以下代碼所示。
public class Enrollment{
public int EnrollmentID { get; set; }
public int CourseID { get; set; }
public int StudentID { get; set; }
public Grade? Grade { get; set; }
public virtual Course Course { get; set; }
[ForeignKey("StudentID")]
public virtual Student Student { get; set; }
}
現在可以看到ForeignKey
屬性應用於導航屬性。
12. NotMapped
Code First的約定在默認情況下,每個屬於受支持數據類型的屬性都包含getter
和setter
,它們在資料庫中表示。 但是在應用中並不總是這樣。 NotMapped
屬性將覆蓋此默認約定。 例如,可能在Student
類中有一個屬性,例如FatherName
,但不需要存儲它到資料庫表。 那麼可以將NotMapped
屬性應用於您不希望在資料庫中創建列的FatherName
屬性。 以下是代碼。
public class Student{
[Key]
public int StdntID { get; set; }
public string LastName { get; set; }
public string FirstMidName { get; set; }
public DateTime EnrollmentDate { get; set; }
[NotMapped]
public int FatherName { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
可以看到NotMapped
屬性應用於FatherName
屬性。 現在,當生成表時,將看到FatherName
列不會在資料庫中創建,但它存在於Student
類中。
Code First 不會為沒有getter
或setter
的屬性創建一個列。
13. InverseProperty
在類之間有多個關係時使用InverseProperty
。 在Enrollment
類中,可能想要跟蹤註冊“當前課程”的人員和註冊“以前課程”的人員。
我們為Enrollment
類添加兩個導航屬性。
public class Enrollment{
public int EnrollmentID { get; set; }
public int CourseID { get; set; }
public int StudentID { get; set; }
public Grade? Grade { get; set; }
public virtual Course CurrCourse { get; set; }
public virtual Course PrevCourse { get; set; }
public virtual Student Student { get; set; }
}
同樣,還需要添加這些屬性引用Course
類。 Course
類的導航屬性返回到Enrollment
類,其中包含當前和以前的所有註冊。
public class Course{
public int CourseID { get; set; }
public string Title { get; set; }
[Index]
public int Credits { get; set; }
public virtual ICollection<Enrollment> CurrEnrollments { get; set; }
public virtual ICollection<Enrollment> PrevEnrollments { get; set; }
}
如果外鍵屬性未包含在上述類中所示的特定類中,Code First會創建{Class Name} _ {Primary Key}
外鍵列。 生成資料庫表時,您將看到許多外鍵,如以下螢幕截圖所示。
正如所看到的Code First 無法自己匹配兩個類的屬性。 用於Enrollments
的資料庫表應該有一個用於CurrCourse
的外鍵和一個用於PrevCourse
的外鍵,但Code First 將創建四個外鍵屬性,即 -
- CurrCourse_CourseID
- PrevCourse_CourseID
- Course_CourseID
- Course_CourseID1
要解決這些問題,可以使用InverseProperty
注解來指定屬性的對齊方式。
public class Course{
public int CourseID { get; set; }
public string Title { get; set; }
[Index]
public int Credits { get; set; }
[InverseProperty("CurrCourse")]
public virtual ICollection<Enrollment> CurrEnrollments { get; set; }
[InverseProperty("PrevCourse")]
public virtual ICollection<Enrollment> PrevEnrollments { get; set; }
}
正如上面所看到的那樣,當InverseProperty
屬性通過指定它所屬的Enrollment
類的哪個引用屬性應用於上述Course
類時,Code First將生成資料庫表,並在Enrollments
表中創建兩個外鍵列,如以下螢幕截圖所示。
我們建議執行上述示例以便更好地理解。