بررسی رابطهی One-to-Zero-or-One
زمانیکه نیاز است موجودیت A با هیچ و یا حداکثر یک وهله از موجودیت B در ارتباط باشد، به یک چنین رابطهای One-to-Zero-or-One میگویند. برای اینکه یک چنین ارتباطی را تشکیل دهیم، نیاز است کلید اصلی یک جدول، در جدول مرتبط به آن، هم به عنوان کلید اصلی و هم به عنوان کلید خارجی معرفی شود؛ همانند شکل زیر که در آن CartableId، همزمان به صورت PK و FK تعریف شدهاست که به آن one-to-one association with shared primary key نیز میگویند:
الف) مدلسازی رابطهی One-to-Zero-or-One توسط Fluent API
در اینجا دو موجودیت را ملاحظه میکنید که توسط دو navigation property به هم متصل شدهاند:
public class UserProfile
{
public int UserProfileId { get; set; }
public string UserName { get; set; }
public virtual Cartable Cartable { get; set; }
}
public class Cartable
{
public int CartableId { get; set; }
public virtual UserProfile UserProfile { get; set; }
}
برای اینکه بتوان CartableId را هم به عنوان PK و هم FK معرفی کرد، نیاز است از Fluent API به نحو ذیل استفاده کنیم:
public class MyDBDataContext : DbContext
{
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.UseSqlServer(@"Data Source=(local);Initial Catalog=testdb2;Integrated Security = true");
}
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<Cartable>(entity =>
{
entity.Property(e => e.CartableId).ValueGeneratedNever();
entity.HasOne(d => d.UserProfile)
.WithOne(p => p.Cartable)
.HasForeignKey<Cartable>(d => d.CartableId);
});
}
public virtual DbSet<Cartable> Cartables { get; set; }
public virtual DbSet<UserProfile> UserProfiles { get; set; }
}
در اینجا رابطهی یک به یک، توسط متدهای HasOne و WithOne معرفی شدهاست. به علاوه FK بودن CartableId به صورت صریح توسط متد HasForeignKey نیز مشخص گردیدهاست.
همچنین بر روی CartableId، فراخوانی متد ValueGeneratedNever را مشاهده میکنید. این متد را در قسمت «روشهای مختلف تولید خودکار مقادیر خواص» مطلب «
شروع به کار با EF Core 1.0 - قسمت 5 - استراتژهای تعیین کلید اصلی جداول و ایندکسها» پیشتر بررسی کردیم. هدف آن این است که به بانک اطلاعاتی اعلام کند، این فیلد یک کلید اصلی از نوع خود افزایش یابنده نیست و مقدار آن توسط برنامه به صورت صریح تنظیم میشود (چون کلید خارجی نیز هست و به کلید اصلی جدول سمت دیگر رابطه اشاره میکند).
ب) مدلسازی رابطهی One-to-Zero-or-One توسط Data Annotations
برای تنظیم این رابطه توسط ویژگیها نیاز است DatabaseGenerated به None تنظیم شود تا کلید اصلی CartableId به صورت خودکار توسط بانک اطلاعاتی تولید نشود. همچنین این کلید اصلی نیز باید به صورت کلید خارجی نیز معرفی شود. به علاوه توسط InversePropertyها، باید هر دو سطر به هم مرتبط، ذکر شوند:
public class Cartable
{
[DatabaseGenerated(DatabaseGeneratedOption.None)]
public int CartableId { get; set; }
[ForeignKey("CartableId")]
[InverseProperty("Cartable")]
public virtual UserProfile UserProfile { get; set; }
}
public class UserProfile
{
public int UserProfileId { get; set; }
public string UserName { get; set; }
[InverseProperty("UserProfile")]
public virtual Cartable Cartable { get; set; }
}
بررسی رابطهی One-to-One
تشکیل رابطهی One-to-One که در آن برخلاف رابطهی One-to-Zero-or-One، وجود هر دو سر رابطه اجباری هستند، در SQL Server میسر نیست (زیرا زمانیکه یک چنین رابطهای تشکیل شود، نمیتوان رکوردی را Insert کرد! چون زمانیکه هنوز یک سر رابطه ثبت نشدهاست، چگونه میتوان Id آنرا در سر دیگری به اجبار ثبت کرد؟!). SQL Server این رابطه را نیز به صورت همان One-to-Zero-or-One تشکیل میدهد. تنظیمات آن نیز با قبل تفاوتی ندارد. در این حالت اجباری بودن و یا نبودن یک سر رابطه همانند نکات قسمت «تعیین اجباری بودن یا نبودن ستونها در EF Core» در مطلب «
شروع به کار با EF Core 1.0 - قسمت 6 - تعیین نوعهای داده و ویژگیهای آنها» است و این تنظیمات در اینجا صرفا از دیدگاه EF Core مفهوم دارند.
جهت تکمیل بحث، روش تشکیل رابطهی One-to-One بدون استفاده از روش به اشتراک گذاری کلید اصلی (one-to-one association with shared primary key) به صورت زیر است:
همانطور که مشاهده میکنید، در اینجا هر بلاگ حداکثر یک تصویر را میتواند داشته باشد. علت آن نیز به ذکر MyBlogForeignKey بر میگردد که یک کلید خارجی نال نپذیر است.
public class MyBlog
{
public int MyBlogId { get; set; }
public string Url { get; set; }
public MyBlogImage MyBlogImage { get; set; }
}
public class MyBlogImage
{
public int MyBlogImageId { get; set; }
public byte[] Image { get; set; }
public string Caption { get; set; }
public int MyBlogForeignKey { get; set; }
public MyBlog MyBlog { get; set; }
}
با این تنظیمات:
modelBuilder.Entity<MyBlog>()
.HasOne(p => p.MyBlogImage)
.WithOne(i => i.MyBlog)
.HasForeignKey<MyBlogImage>(b => b.MyBlogForeignKey);
در اینجا جدول MyBlogImage هنوز Id خود افزایندهی خود را دارد، اما ثبت رکورد آن بدون وجود کلید خارجی MyBlog میسر نیست.