یکی از روشهای Join نوشتن درEF ، استفاده از Includeها است. اما ... آیا تمام Includeهایی که تعریف شدهاند ضروری بودهاند؟ آیا تمام Joinهای حاصل از Includeهای تعریف شده مورد استفاده قرار گرفتهاند و بیجهت کارآیی برنامه را پایین نیاوردهاند؟ EF Core برای مقابله با یک چنین مواردی که بسیار هم متداول هستند، پس از بررسی کوئری، سعی میکند Includeهای اضافی و بیمصرف را حذف کرده و سبب تولید Joinهای اضافی نشود.
یک مثال: بررسی تنظیمات نمایش خطاهای Includeهای صرفنظر شده
موجودیتهای Blog و Postهای آنرا درنظر بگیرید:
public class Blog
{
public int BlogId { get; set; }
public string Url { get; set; }
public List<Post> Posts { get; set; }
}
public class Post
{
public int PostId { get; set; }
public string Title { get; set; }
public string Content { get; set; }
public int BlogId { get; set; }
public Blog Blog { get; set; }
}
به همراه Context آن که به صورت ذیل تعریف شدهاست:
public class BloggingContext : DbContext
{
public BloggingContext()
{ }
public BloggingContext(DbContextOptions options)
: base(options)
{ }
public DbSet<Blog> Blogs { get; set; }
public DbSet<Post> Posts { get; set; }
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
if (!optionsBuilder.IsConfigured)
{
optionsBuilder.UseSqlServer(@"Server=(localdb)\mssqllocaldb;Database=Demo.Includes;Trusted_Connection=True;");
optionsBuilder.ConfigureWarnings(warnings => warnings.Log(CoreEventId.IncludeIgnoredWarning));
optionsBuilder.UseLoggerFactory(new LoggerFactory().AddConsole((message, logLevel) =>
{
return true;
/*return logLevel == LogLevel.Debug &&
message.StartsWith("Microsoft.EntityFrameworkCore.Database.Command");*/
}));
}
}
}
در اینجا در تنظیمات ConfigureWarnings، نمایش IncludeIgnoredWarning نیز لحاظ شدهاست. به این ترتیب میتوان از وقوع Includeهای صرفنظر شده مطلع شد.
همچنین ذکر UseLoggerFactory، به نحوی که مشاهده میکنید، یکی دیگر از روشهای لاگ کردن خروجیهای EF است. در اینجا اگر true تنظیم شود، تمام اتفاقات مرتبط با EF Core لاگ میشوند. اگر میخواهید تنها کوئریهای SQL آنرا مشاهده کنید، روش فیلتر کردن آنها در سطر بعدی نمایش داده شدهاست.
در این حالت اگر کوئری ذیل را اجرا کنیم:
var blogs = context.Blogs
.Include(blog => blog.Posts)
.Select(blog => new
{
Id = blog.BlogId,
Url = blog.Url
})
.ToList();
ابتدا خطای ذیل نمایش داده میشود:
warn: Microsoft.EntityFrameworkCore.Query[100106]
The Include operation for navigation '[blog].Posts' is unnecessary and was ignored because the navigation is not reachable in the final query results. See https://go.microsoft.com/fwlink/?linkid=850303 for more information.
سپس کوئری نهایی تشکیل شده نیز به صورت ذیل است:
SELECT [blog].[BlogId] AS [Id], [blog].[Url]
FROM [Blogs] AS [blog]
همانطور که ملاحظه میکنید در اینجا خبری از join به جدول posts نیست و یک کوئری ساده را تشکیل دادهاست. چون خروجی نهایی که در Select قید شدهاست، ارتباطی به جدول posts ندارد. به همین جهت، تشکیل join بر روی آن غیرضروری است و با این خطای ذکر شده میتوان دریافت که بهتر است Include ذکر شده را از کوئری حذف کرد.
یک نکته: اگر تنظیم نمایش IncludeIgnoredWarning را به نحو ذیل به Throw تنظیم کنیم:
optionsBuilder.ConfigureWarnings(warnings => warnings.Throw(CoreEventId.IncludeIgnoredWarning));
اینبار برنامه با رسیدن به یک چنین کوئریهایی، متن اخطار ذکر شده را به صورت یک استثناء صادر خواهد کرد. این حالت در زمان توسعهی برنامه میتواند مفید باشد.
کدهای کامل این مثال را از اینجا میتوانید دریافت کنید: IgnoredIncludes.zip