Entity Framework-데이터 주석
DataAnnotations는 가장 일반적으로 필요한 구성을 강조하는 클래스를 구성하는 데 사용됩니다. DataAnnotations는 ASP.NET MVC와 같은 여러 .NET 응용 프로그램에서도 이해되며 이러한 응용 프로그램은 클라이언트 측 유효성 검사에 동일한 주석을 활용할 수 있습니다. DataAnnotation 속성은 기본 CodeFirst 규칙을 재정의합니다.
System.ComponentModel.DataAnnotations Null 허용 여부 또는 열 크기에 영향을주는 다음 속성을 포함합니다.
- Key
- Timestamp
- ConcurrencyCheck
- Required
- MinLength
- MaxLength
- StringLength
System.ComponentModel.DataAnnotations.Schema 네임 스페이스에는 데이터베이스의 스키마에 영향을주는 다음 속성이 포함됩니다.
- Table
- Column
- Index
- ForeignKey
- NotMapped
- InverseProperty
키
Entity Framework는 엔터티 추적에 사용하는 키 값이있는 모든 엔터티에 의존합니다. Code First가 의존하는 규칙 중 하나는 각 Code First 클래스에서 어떤 속성이 키인지를 의미하는 방식입니다.
규칙은 "Id"라는 속성 또는 "StudentId"와 같이 클래스 이름과 "Id"를 결합한 속성을 찾는 것입니다.
속성은 데이터베이스의 기본 키 열에 매핑됩니다.
학생, 과정 및 등록 수업은이 규칙을 따릅니다.
이제 Student 클래스가 ID 대신 StdntID라는 이름을 사용했다고 가정 해 보겠습니다. Code First가이 규칙과 일치하는 속성을 찾지 못하면 키 속성이 있어야한다는 Entity Framework의 요구 사항으로 인해 예외가 발생합니다. 키 주석을 사용하여 EntityKey로 사용할 속성을 지정할 수 있습니다.
StdntID를 포함하지만 기본 Code First 규칙을 따르지 않는 Student 클래스의 다음 코드를 살펴 보겠습니다. 따라서이를 처리하기 위해 키 속성이 추가되어 기본 키가됩니다.
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 Explorer에서 데이터베이스를 살펴보면 이제 기본 키가 Students 테이블의 StdntID임을 알 수 있습니다.
Entity Framework는 복합 키도 지원합니다. Composite keys둘 이상의 속성으로 구성된 기본 키이기도합니다. 예를 들어 기본 키가 LicenseNumber 및 IssuingCountry의 조합 인 DrivingLicense 클래스가 있습니다.
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; }
}
복합 키가있는 경우 Entity Framework에서는 키 속성의 순서를 정의해야합니다. 열 주석을 사용하여 순서를 지정할 수 있습니다.
타임 스탬프
Code First는 Timestamp 속성을 ConcurrencyCheck 속성과 동일하게 처리하지만 코드가 처음 생성하는 데이터베이스 필드가 null이 아닌지 확인합니다.
동시성 검사를 위해 rowversion 또는 timestamp 필드를 사용하는 것이 더 일반적입니다.
ConcurrencyCheck 주석을 사용하는 대신 속성 유형이 바이트 배열 인 한보다 구체적인 TimeStamp 주석을 사용할 수 있습니다.
주어진 클래스에는 하나의 타임 스탬프 속성 만있을 수 있습니다.
Course 클래스에 TimeStamp 속성을 추가하여 간단한 예를 살펴 보겠습니다.
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는 TStamp
Courses 테이블에 타임 스탬프 열 을 생성 합니다.
동시성 검사
ConcurrencyCheck 주석을 사용하면 사용자가 항목을 편집하거나 삭제할 때 데이터베이스에서 동시성 검사에 사용할 하나 이상의 속성에 플래그를 지정할 수 있습니다. EF 디자이너로 작업 한 경우 속성의 ConcurrencyMode를 고정으로 설정하는 것과 일치합니다.
ConcurrencyCheck를 Course 클래스의 Title 속성에 추가하여 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; }
}
위 과정 클래스에서 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
필수 주석
Required 주석은 특정 속성이 필요하다는 것을 EF에 알려줍니다. FirstMidName 속성에 Required id가 추가 된 다음 Student 클래스를 살펴 보겠습니다. 필수 특성은 EF가 속성에 데이터가 있는지 확인하도록합니다.
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; }
}
위의 예에서 볼 수 있듯이 Required 속성은 FirstMidName 및 LastName에 적용됩니다. 따라서 Code First는 다음 이미지와 같이 Students 테이블에 NOT NULL FirstMidName 및 LastName 열을 만듭니다.
MaxLength
MaxLength 속성을 사용하면 추가 속성 유효성 검사를 지정할 수 있습니다. 도메인 클래스의 문자열 또는 배열 유형 속성에 적용 할 수 있습니다. EF Code First는 MaxLength 특성에 지정된대로 열의 크기를 설정합니다.
Title 속성에 MaxLength (24) 속성이 적용된 다음 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는 다음 이미지와 같이 CourseId 테이블에 nvarchar (24) 열 제목을 만듭니다.
사용자가 24 자 이상의 제목을 설정하면 EF는 EntityValidationError를 발생시킵니다.
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 특성에 지정된 길이보다 작거나 MaxLength 특성에 지정된 길이보다 큰 Title 속성 값을 설정하면 EF에서 EntityValidationError를 발생시킵니다.
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 특성에 대한 속성 값의 유효성을 검사합니다. 사용자가 24자를 초과하는 제목을 설정하면 EF는 EntityValidationError를 발생시킵니다.
표
기본 Code First 규칙은 클래스 이름과 유사한 테이블 이름을 만듭니다. Code First가 데이터베이스를 생성하도록하고 생성중인 테이블의 이름도 변경하려는 경우. 그런 다음-
기존 데이터베이스에서 Code First를 사용할 수 있습니다. 그러나 클래스 이름이 데이터베이스의 테이블 이름과 항상 일치하는 것은 아닙니다.
테이블 속성은이 기본 규칙을 재정의합니다.
EF Code First는 주어진 도메인 클래스에 대해 Table 특성에 지정된 이름으로 테이블을 만듭니다.
클래스 이름이 Student이고 규칙에 따라 Code First는이 클래스가 Students라는 테이블에 매핑되는 것으로 가정하는 다음 예제를 살펴 보겠습니다. 그렇지 않은 경우 다음 코드와 같이 Table 속성을 사용하여 테이블 이름을 지정할 수 있습니다.
[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로 지정하는 것을 볼 수 있습니다. 테이블이 생성되면 다음 이미지와 같이 테이블 이름 StudentsInfo가 표시됩니다.
테이블 이름을 지정할 수있을뿐만 아니라 다음 코드와 같이 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; }
}
위의 예에서 테이블이 관리 스키마로 지정되었음을 알 수 있습니다. 이제 Code First는 다음 이미지와 같이 Admin 스키마에 StudentsInfo 테이블을 생성합니다.
기둥
또한 Table 속성과 동일하지만 Table 속성은 테이블 동작을 재정의하고 Column 속성은 열 동작을 재정의합니다. 기본 Code First 규칙은 속성 이름과 유사한 열 이름을 만듭니다. Code First가 데이터베이스를 생성하도록하고 테이블의 열 이름도 변경하려는 경우. 그런 다음-
열 속성은 기본 규칙을 재정의합니다.
EF Code First는 지정된 속성에 대한 열 특성에 지정된 이름으로 열을 만듭니다.
속성의 이름이 FirstMidName이고 규칙에 따라 Code First는 이것이 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이 표시됩니다.
인덱스
Index 특성은 Entity Framework 6.1에서 도입되었습니다. 이전 버전을 사용하는 경우이 섹션의 정보가 적용되지 않습니다.
IndexAttribute를 사용하여 하나 이상의 열에 인덱스를 만들 수 있습니다.
하나 이상의 속성에 특성을 추가하면 EF는 데이터베이스를 만들 때 데이터베이스에 해당 인덱스를 만듭니다.
인덱스를 사용하면 대부분의 경우 데이터를 더 빠르고 효율적으로 검색 할 수 있습니다. 그러나 인덱스가있는 테이블 또는 뷰를 오버로드하면 삽입 또는 업데이트와 같은 다른 작업의 성능에 좋지 않은 영향을 미칠 수 있습니다.
인덱싱은 데이터베이스에서 데이터를 쿼리하는 데 필요한 시간을 줄여 Code First 애플리케이션의 성능을 향상시킬 수있는 Entity Framework의 새로운 기능입니다.
Index 속성을 사용하여 데이터베이스에 인덱스를 추가하고 기본 고유 및 클러스터 설정을 재정 의하여 시나리오에 가장 적합한 인덱스를 얻을 수 있습니다.
기본적으로 인덱스 이름은 IX_ <property name>입니다.
학점에 대한 Course 클래스에 Index 속성이 추가 된 다음 코드를 살펴 보겠습니다.
public class Course {
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; }
}
ForeignKey
Code First 규칙은 모델에서 가장 일반적인 관계를 처리하지만 도움이 필요한 경우가 있습니다. 예를 들어 Student 클래스에서 키 속성의 이름을 변경하면 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 속성이 없지만 StdntID 속성은 Student 클래스입니다.
이에 대한 해결책은 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 특성이 탐색 속성에 적용되었음을 알 수 있습니다.
NotMapped
기본적으로 Code First 규칙에 따라 지원되는 데이터 유형이고 getter 및 setter를 포함하는 모든 속성이 데이터베이스에 표시됩니다. 그러나 이것은 귀하의 응용 프로그램에서 항상 그런 것은 아닙니다. NotMapped 속성은이 기본 규칙을 재정의합니다. 예를 들어, FatherName과 같은 Student 클래스에 속성이있을 수 있지만 저장할 필요는 없습니다. 다음 코드와 같이 데이터베이스에 열을 생성하지 않으려는 FatherName 속성에 NotMapped 특성을 적용 할 수 있습니다.
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는 Student 클래스의 Address 및 Age 속성에 대한 다음 예제와 같이 getter 또는 setter가없는 속성에 대한 열을 만들지 않습니다.
InverseProperty
InverseProperty는 클래스간에 여러 관계가있을 때 사용됩니다. 등록 클래스에서 현재 코스와 이전 코스를 등록한 사람을 추적 할 수 있습니다. 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는 자체적으로 두 클래스의 속성을 일치시킬 수 없습니다. 등록 용 데이터베이스 테이블에는 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는 다음 이미지와 같이 데이터베이스를 생성하고 등록 테이블에 외래 키 열을 두 개만 만듭니다.
더 나은 이해를 위해 위의 예를 단계별로 실행하는 것이 좋습니다.