Kerangka Kerja Entitas - Contoh Pertama
Mari kita definisikan model yang sangat sederhana menggunakan kelas. Kami hanya mendefinisikannya di file Program.cs tetapi dalam aplikasi dunia nyata Anda akan membagi kelas Anda menjadi file terpisah dan berpotensi menjadi proyek terpisah. Berikut model data yang akan kita buat menggunakan pendekatan Code First.
Buat Model
Tambahkan tiga kelas berikut dalam file Program.cs menggunakan kode berikut untuk kelas Siswa.
public class Student {
public int ID { get; set; }
public string LastName { get; set; }
public string FirstMidName { get; set; }
public DateTime EnrollmentDate { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
Properti ID akan menjadi kolom kunci utama dari tabel database yang sesuai dengan kelas ini.
Properti Pendaftaran adalah properti navigasi. Properti navigasi menyimpan entitas lain yang terkait dengan entitas ini.
Dalam kasus ini, properti Pendaftaran entitas Mahasiswa akan menampung semua entitas Pendaftaran yang terkait dengan entitas Mahasiswa tersebut.
Properti navigasi biasanya didefinisikan sebagai virtual sehingga dapat memanfaatkan fungsionalitas Kerangka Kerja Entitas tertentu seperti pemuatan lambat.
Jika properti navigasi bisa menampung banyak entitas (seperti dalam hubungan banyak-ke-banyak atau satu-tomany), tipenya harus berupa daftar di mana entri dapat ditambahkan, dihapus, dan diperbarui, seperti ICollection.
Berikut implementasi untuk kelas Course.
public class Course {
public int CourseID { get; set; }
public string Title { get; set; }
public int Credits { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
Properti Pendaftaran adalah properti navigasi. Entitas Kursus dapat dikaitkan dengan sejumlah entitas Pendaftaran.
Berikut implementasi untuk Enrollment class dan enum.
public enum Grade {
A, B, C, D, F
}
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; }
}
Properti EnrollmentID akan menjadi kunci utama.
Properti Grade adalah enum. Tanda tanya setelah deklarasi tipe Grade menunjukkan bahwa properti Grade nullable.
Nilai yang nol berbeda dengan nilai nol. Null berarti nilai belum diketahui atau belum ditetapkan.
Properti StudentID dan CourseID adalah kunci asing, dan properti navigasi terkait adalah Student dan Course.
Entitas Pendaftaran dikaitkan dengan satu Siswa dan satu entitas Kursus, sehingga properti hanya dapat menampung satu entitas Siswa dan Kursus.
Buat Konteks Database
Kelas utama yang mengoordinasikan fungsionalitas Entity Framework untuk model data tertentu adalah kelas konteks database yang memungkinkan untuk melakukan kueri dan menyimpan data. Anda dapat membuat kelas ini dengan mengambil dari kelas DbContext dan mengekspos DbSet yang diketik
public class MyContext : DbContext {
public virtual DbSet<Course> Courses { get; set; }
public virtual DbSet<Enrollment> Enrollments { get; set; }
public virtual DbSet<Student> Students { get; set; }
}
Berikut kode lengkap di file Program.cs.
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace EFCodeFirstDemo {
class Program {
static void Main(string[] args) {}
}
public enum Grade {
A, B, C, D, F
}
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 {
public int ID { get; set; }
public string LastName { get; set; }
public string FirstMidName { get; set; }
public DateTime EnrollmentDate { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
public class Course {
public int CourseID { get; set; }
public string Title { get; set; }
public int Credits { get; set; }
public virtual ICollection<Enrollment> Enrollments { get; set; }
}
public class MyContext : DbContext {
public virtual DbSet<Course> Courses { get; set; }
public virtual DbSet<Enrollment> Enrollments { get; set; }
public virtual DbSet<Student> Students { get; set; }
}
}
Kode di atas adalah semua yang kita butuhkan untuk mulai menyimpan dan mengambil data. Mari tambahkan beberapa data dan kemudian ambil kembali. Berikut adalah kode dalam metode utama.
static void Main(string[] args) {
using (var context = new MyContext()) {
// Create and save a new Students
Console.WriteLine("Adding new students");
var student = new Student {
FirstMidName = "Alain", LastName = "Bomer",
EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
};
context.Students.Add(student);
var student1 = new Student {
FirstMidName = "Mark", LastName = "Upston",
EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
};
context.Students.Add(student1);
context.SaveChanges();
// Display all Students from the database
var students = (from s in context.Students
orderby s.FirstMidName select s).ToList<Student>();
Console.WriteLine("Retrieve all Students from the database:");
foreach (var stdnt in students) {
string name = stdnt.FirstMidName + " " + stdnt.LastName;
Console.WriteLine("ID: {0}, Name: {1}", stdnt.ID, name);
}
Console.WriteLine("Press any key to exit...");
Console.ReadKey();
}
}
Ketika kode di atas dijalankan, Anda akan menerima output berikut.
Adding new students
Retrieve all Students from the database:
ID: 1, Name: Alain Bomer
ID: 2, Name: Mark Upston
Press any key to exit...
Sekarang pertanyaan yang muncul di benak adalah, di mana data dan database di mana kita telah menambahkan beberapa data dan kemudian mengambilnya dari database. Sesuai kesepakatan, DbContext telah membuat database untuk Anda.
Jika contoh SQL Express lokal tersedia maka Kode Pertama telah membuat database pada contoh itu.
Jika SQL Express tidak tersedia, maka Kode Pertama akan mencoba dan menggunakan LocalDb.
Database dinamai sesuai nama yang memenuhi syarat dari konteks turunan.
Dalam kasus kami, contoh SQL Express tersedia, dan nama database adalah EFCodeFirstDemo.MyContext seperti yang ditunjukkan pada gambar berikut.
Ini hanyalah konvensi default dan ada berbagai cara untuk mengubah database yang digunakan Code First.
Seperti yang Anda lihat pada gambar di atas, ini telah membuat tabel Siswa, Kursus dan Pendaftaran dan setiap tabel berisi kolom dengan tipe data dan panjang yang sesuai.
Nama kolom dan tipe data juga cocok dengan properti kelas domain masing-masing.
Inisialisasi Database
Pada contoh di atas, kita telah melihat bahwa Code First membuat database secara otomatis, tetapi jika Anda ingin mengubah nama database dan server, mari kita lihat bagaimana Code First memutuskan nama database dan server saat menginisialisasi database. Perhatikan diagram berikut.
Anda dapat menentukan konstruktor dasar dari kelas konteks dengan cara berikut.
- Tidak Ada Parameter
- Nama Database
- Nama String Koneksi
Tidak Ada Parameter
Jika Anda menentukan konstruktor dasar kelas konteks tanpa parameter apa pun seperti yang ditunjukkan pada contoh di atas, maka kerangka kerja entitas akan membuat database di server SQLEXPRESS lokal Anda dengan nama {Namespace}. {Nama kelas konteks}.
Pada contoh di atas, database yang dibuat secara otomatis memiliki nama EFCodeFirstDemo.MyContext. Jika Anda melihat namanya, Anda akan menemukan bahwa EFCodeFirstDemo adalah namespace dan MyContext adalah nama kelas konteks seperti yang ditunjukkan pada kode berikut.
public class MyContext : DbContext {
public MyContext() : base() {}
public virtual DbSet<Course> Courses { get; set; }
public virtual DbSet<Enrollment> Enrollments { get; set; }
public virtual DbSet<Student> Students { get; set; }
}
Nama Database
Jika Anda meneruskan nama database sebagai parameter di konstruktor dasar kelas konteks, maka Kode Pertama akan membuat database secara otomatis lagi, tetapi kali ini nama akan menjadi yang diteruskan sebagai parameter di konstruktor dasar di server database SQLEXPRESS lokal .
Dalam kode berikut, MyContextDB ditetapkan sebagai parameter di konstruktor dasar. Jika menjalankan aplikasi Anda, maka database dengan nama MyContextDB akan dibuat di server SQL lokal Anda.
public class MyContext : DbContext {
public MyContext() : base("MyContextDB") {}
public virtual DbSet<Course> Courses { get; set; }
public virtual DbSet<Enrollment> Enrollments { get; set; }
public virtual DbSet<Student> Students { get; set; }
}
Nama String Koneksi
Ini adalah cara mudah untuk memberi tahu DbContext untuk menggunakan server database selain SQL Express atau LocalDb. Anda dapat memilih untuk meletakkan string koneksi di file app.config Anda.
Jika nama string koneksi cocok dengan nama konteks Anda (baik dengan atau tanpa kualifikasi namespace), maka itu akan ditemukan oleh DbContext ketika parameter less constructor digunakan.
Jika nama string koneksi berbeda dari nama konteks Anda, maka Anda dapat memberi tahu DbContext untuk menggunakan koneksi ini dalam mode Code First dengan meneruskan nama string koneksi ke konstruktor DbContext.
public class MyContext : DbContext {
public MyContext() : base("name = MyContextDB") {}
public virtual DbSet<Course> Courses { get; set; }
public virtual DbSet<Enrollment> Enrollments { get; set; }
public virtual DbSet<Student> Students { get; set; }
}
Dalam kode di atas, potongan string koneksi kelas konteks ditentukan sebagai parameter dalam konstruktor dasar.
Nama string koneksi harus dimulai dengan "name =" jika tidak, ini akan menganggapnya sebagai nama database.
Formulir ini menjelaskan bahwa Anda mengharapkan string koneksi ditemukan di file konfigurasi Anda. Pengecualian akan diberikan jika string koneksi dengan nama yang diberikan tidak ditemukan.
<connectionStrings>
<add name = "MyContextDB"
connectionString = "Data Source =.;Initial Catalog = EFMyContextDB;Integrated Security = true"
providerName = "System.Data.SqlClient"/>
</connectionStrings>
Nama database dalam string koneksi di app.config adalah EFMyContextDB. CodeFirst akan membuat fileEFMyContextDB database atau gunakan yang sudah ada EFMyContextDB database di SQL Server lokal.
Kelas Domain
Sejauh ini kita baru saja membiarkan EF menemukan model menggunakan konvensi default, tetapi akan ada saatnya kelas kita tidak mengikuti konvensi dan kita perlu melakukan konfigurasi lebih lanjut. Tetapi Anda dapat mengesampingkan konvensi ini dengan mengonfigurasi kelas domain Anda untuk memberikan informasi yang dibutuhkan EF. Ada dua opsi untuk mengonfigurasi kelas domain Anda -
- Anotasi Data
- API Lancar
Anotasi Data
DataAnnotations digunakan untuk mengonfigurasi kelas Anda yang akan menyoroti konfigurasi yang paling sering dibutuhkan. DataAnnotations juga dipahami oleh sejumlah aplikasi .NET, seperti ASP.NET MVC yang memungkinkan aplikasi ini memanfaatkan anotasi yang sama untuk validasi sisi klien.
Berikut adalah data anotasi yang digunakan di kelas siswa.
public class Enrollment {
[Key]
public int EnrollmentID { get; set; }
public int CourseID { get; set; }
public int StudentID { get; set; }
public Grade? Grade { get; set; }
[ForeignKey("CourseID")]
public virtual Course Course { get; set; }
[ForeignKey("ID")]
public virtual Student Student { get; set; }
}
API Lancar
Sebagian besar konfigurasi model dapat dilakukan menggunakan anotasi data sederhana. Fluent API adalah cara lanjutan untuk menentukan konfigurasi model yang mencakup semua hal yang dapat dilakukan anotasi data, selain beberapa konfigurasi lanjutan yang tidak dapat dilakukan dengan anotasi data. Anotasi data dan fluent API dapat digunakan bersama.
Untuk mengakses fluent API Anda mengganti metode OnModelCreating di DbContext. Sekarang mari kita ganti nama kolom pada tabel siswa dari FirstMidName menjadi FirstName seperti yang ditunjukkan pada kode berikut.
public class MyContext : DbContext {
protected override void OnModelCreating(DbModelBuilder modelBuilder) {
modelBuilder.Entity<Student>().Property(s ⇒ s.FirstMidName)
.HasColumnName("FirstName");
}
public virtual DbSet<Course> Courses { get; set; }
public virtual DbSet<Enrollment> Enrollments { get; set; }
public virtual DbSet<Student> Students { get; set; }
}