یک نکتهی تکمیلی: تاثیر نوعهای ارجاعی نال نپذیر C# 8.0 بر روی EF Core 3.0
تغییرات نحوهی تعریف موجودیتها در C# 8.0
تا پیش از C# 8.0، برای تعریف فیلدهای نال نپذیر و نال پذیر در موجودیتهای EF Core، به صورت زیر عمل میشد:
اگر رشتهای مزین به ویژگی Required باشد، یعنی به یک فیلد نالنپذیر، ترجمه و نگاشت خواهد شد و برعکس. اما پس از فعالسازی ویژگی نوعهای ارجاعی نال نپذیر C# 8.0 در پروژهی خود، کامپایلر اخطارهایی مانند «Non-nullable property 'FirstName' is uninitialized. Consider declaring the property as nullable» را به ازای تک تک خواص تعریف شدهی در کلاس موجودیت فوق، صادر میکند. برای رفع این مشکل میتوان از bang operator که کمی بالاتر در مورد آن توضیح داده شده، استفاده کرد:
در اینجا نحوهی تعریف دو فیلد نالنپذیر و نال پذیر را در موجودیتهای EF Core 3.0 و C# 8.0 مشاهده میکنید. خاصیت اول دیگر نیازی به ویژگی Required ندارد؛ اما چون دیگر نال را نمیپذیرد، میتوان مقدار دهی اولیهی آنرا توسط !null انجام داد؛ تا کامپایلر دیگر خطایی را در مورد عدم مقدار دهی اولیهی آن صادر نکند (تنها کاربرد !null است). البته بهتر است !null را صرفا با EF Core و موجودیتهای آن استفاده کنید و برای سایر کلاسها، از دیگر روشهای مطرح شدهی در این مطلب مانند تعریف یک سازنده، کمک بگیرید.
مزیت این روش نسبت به Person_BeforeCS8 این است که اینبار علاوه بر نالنپذیر تعریف شدن فیلد FirstName، نحوهی استفادهی از آن در برنامه (عدم انتساب نال به آن) نیز تحت کنترل کامپایلر قرار میگیرد که پیشتر با ویژگی Required چنین امری میسر نبود.
بنابراین در موجودیتهای برنامهی مبتنی بر C# 8.0، دیگر نیاز به استفادهی از ویژگی Required نبوده و نالپذیری با عملگر ? مشخص میشود.
کار با وابستگیها و ارتباطهای نالپذیر
فرض کنید یک چنین کوئری را در EF Core 3.0 و C# 8.0 نوشتهاید:
در اینجا ParentPost میتواند نال باشد، اما در عمل EF Core به این موضوع اهمیتی نمیدهد و از آن صرفا جهت تهیهی SQL نهایی استفاده میکند؛ اما کامپایلر C# 8.0، اخطار «Dereference of a possible null reference» را صادر میکند. برای رفع آن نیز میتوان از bang operator استفاده کرد:
وجود عملگر ! در اینجا، به معنای اعلام صریح نال نبودن ParentPost، در شرایط کوئری فوق، به کامپایلر است.
تغییرات نحوهی تعریف موجودیتها در C# 8.0
تا پیش از C# 8.0، برای تعریف فیلدهای نال نپذیر و نال پذیر در موجودیتهای EF Core، به صورت زیر عمل میشد:
public class Person_BeforeCS8 { [Required] public string FirstName { get; set; } // NOT NULL public string MiddleName { get; set; } // NULL }
public class Person_AfterCS8 { public string FirstName { get; set; } = null!; // NOT NULL public string? MiddleName { get; set; } // NULL }
مزیت این روش نسبت به Person_BeforeCS8 این است که اینبار علاوه بر نالنپذیر تعریف شدن فیلد FirstName، نحوهی استفادهی از آن در برنامه (عدم انتساب نال به آن) نیز تحت کنترل کامپایلر قرار میگیرد که پیشتر با ویژگی Required چنین امری میسر نبود.
بنابراین در موجودیتهای برنامهی مبتنی بر C# 8.0، دیگر نیاز به استفادهی از ویژگی Required نبوده و نالپذیری با عملگر ? مشخص میشود.
کار با وابستگیها و ارتباطهای نالپذیر
فرض کنید یک چنین کوئری را در EF Core 3.0 و C# 8.0 نوشتهاید:
var parentPosts = db.Posts.Where(p => p.ParentPost.Id == postId).ToList();
var parentPosts = db.Posts.Where(p => p.ParentPost!.Id == postId).ToList();