بررسی عبارت Join
عبارت Join برای ایجاد ارتباط بین عناصر دو توالی استفاده میشود. پیش نیاز این عملیات وجود یک مقدار در عناصر توالی ورودی است که بتوان از طریق آن برابری دو عنصر را اهراز کرد.
عبارت join، دو توالی را به عنوان ورودی دریافت میکند. هر عنصر در توالی اول باید یک خصوصیت داشته باشد که بتوان آن را با یک خصوصیت از عناصر توالی دوم مقایسه کرد. عبارت Join، مقادیر مشخص شده را از طریق کلمهی کلیدی equals مقایسه میکند. باید توجه داشت که عملیات join برای بررسی برابری دو مقدار، در دو توالی مختلف است. از این رو به آن equal-joins میگویند.
فرم خروجی عبارت join بر اساس نوع عملیات join ای است که ما انجام میدهیم.
انواع JOIN :
• Inner JOIN
• Group JOIN
• Left JOIN
کلاسهای زیر را در نظر بگیرید:
/// <summary>
/// دستور العمل
/// </summary>
class Recipe
{
public int Id { get; set; }
public string Name { get; set; }
}
/// <summary>
/// بازخورد
/// </summary>
class Review
{
public int RecipeId { get; set; }
public string ReviewText { get; set; }
}
بر اساس مدلهای تعریف شده، هر Recipe ممکن است 0، 1 یا تعداد نامحدودی Review داشته باشد. همانطور که میبینید هیچ رابطهی مستقیمی این دو مدل با هم ندارند. در کلاس Review یک خصوصیت وجود دارد که ID مدل Recipe را در خود نگه میدارد.
Inner Join
این دستور عنصری از توالی اول را که متناظر با آن عنصری در توالی دوم وجود داشته باشد، به خروجی میبرد.
مثال:
Recipe[] recipes =
{
new Recipe {Id = 1, Name = "Mashed Potato"},
new Recipe {Id = 2, Name = "Crispy Duck"},
new Recipe {Id = 3, Name = "Sachertorte"}
};
Review[] reviews =
{
new Review {RecipeId = 1, ReviewText = "Tasty!"},
new Review {RecipeId = 1, ReviewText = "Not nice :("},
new Review {RecipeId = 1, ReviewText = "Pretty good"},
new Review {RecipeId = 2, ReviewText = "Too hard"},
new Review {RecipeId = 2, ReviewText = "Loved it"}
};
var query = from recipe in recipes
join review in reviews on recipe.Id equals review.RecipeId
select new //anonymous type
{
RecipeName = recipe.Name,
RecipeReview = review.ReviewText
};
foreach (var item in query)
{
Console.WriteLine($"{item.RecipeName}-{item.RecipeReview}");
}
در مثال فوق، در ابتدا دو توالی تعریف شد. recipes که تعدادی دستور العمل و reviews که مجموعهای از بازخوردها را در خود نگه میدارند.
پرس و جو با عبارت from که اشارهگری به توالی اول است آغاز و با معرفی یک متغیر Range به نام Recipe ادامه پیدا میکند. سپس دستور Join نوشته شده است. مجددا متغیر range جدیدی به نام review معرفی شده که به عناصر توالی دوم اشاره خواهد کرد. در ادامه کلمهی کلیدی on اجازه میدهد که عناصر یکسانی در توالی اول را با عناصر یکسانی در توالی دوم مشخص کنیم. کلمهی کلیدی equal هم برای مقایسهی برابری دو مقدار استفاده شده است. عبارت join بر اساس مقدار id در مجموعه recipes و همچنین Recipeid در مجموعهی reviews خروجی را مهیا خواهد کرد.
خروج مثال فوق به شکل زیر است :
Mashed Potato-Tasty!
Mashed Potato-Not nice :(
Mashed Potato-Pretty good
Crispy Duck-Too hard
Crispy Duck-Loved it
همانطور که مشاهده میکنید کلیه دستورالعملهایی که بازخوردی برای آنها وجود دارد، در خروجی وجود دارند.
Group Join
بکارگیری into به همراه join، دستور Group Join را میسازد.
این دستور خروجی حاصل از join را گروه بندی خواهد کرد. Group join یک توالی سلسله مراتبی را تولید میکند که در آن یک عنصر از توالی اول با یک یا چند عنصر در توالی دوم در ارتباط است.
مثال:
var query = from recipe in recipes
join review in reviews on recipe.Id equals review.RecipeId
into reviewGroup
select new //anonymous type
{
RecipeName = recipe.Name,
Reviews = reviewGroup//collection of related reviews
};
foreach (var item in query)
{
Console.WriteLine($"Review for {item.RecipeName}");
foreach (var review in item.Reviews)
{
Console.WriteLine($"-{review.ReviewText}");
}
}
در این مثال از 2 توالی تعریف شدهی در مثال قبل استفاده کردهایم.
متغیر reviewGroup توالی حاصل از اجرای join را نمایش میدهد. برای ایجاد توالی خروجی، نتیجه به یک نوع بی نام، بازتاب شده است. هر عنصر در نوع بی نام یک گروه را نشان میدهد. نوع بی نام شامل دو خصوصیت RecipeName که مقدار آن از توالی اول میآید و Reviews که حاصل خروجی Join است میباشد.
خروجی مثال بالا به شکل زیر است:
Review for Mashed Potato
-Tasty!
-Not nice :(
-Pretty good
Review for Crispy Duck
-Too hard
-Loved it
Review for Sachertorte
همانطور که میبینید اینبار عنصر Sachertorte در خروجی ظاهر شده است. اگر عنصری در توالی سمت چپ وجود داشته باشد که عنصر و یا عناصری در توالی سمت راست با آن متناظر نباشند، آنگاه دستور join، یک لیست خالی را برای این عنصر تخصیص میدهد.
Left outer join
برای آنکه بتوان یک خروجی غیر سلسله مراتبی و اصطلاحا flat را ایجاد کرد که مانند خروجی مثال group join باشد (نمایش عنصر Sachertorte) میتوان از عملگر ()DefaultIfEmpty بههمراه یک عبارت from اضافی و معرفی یک متغیر Range استفاده کرد. در صورت نبودن عنصر متناظر در توالی دوم این متغیر range بصورت null تنظیم میشود.
مثال :
در این مثال از مجموعههای تعریف شدهی در بخش اول مطلب استفاده کردهایم:
var query = from recipe in recipes
join review in reviews on recipe.Id equals review.RecipeId
into reviewGroup
from rg in reviewGroup.DefaultIfEmpty()
select new //anonymous type
{
RecipeName = recipe.Name,
//RecipeReview = rg.ReviewText SystemNullException
RecipeReview = (rg == null ? "n/a" : rg.ReviewText)
};
foreach (var item in query)
{
Console.WriteLine($"{item.RecipeName}-{item.RecipeReview}");
}
در بخشی از کد بالا به خاطر وجود حالت null، در خروجی استثنائی رخ خواهد داد. به همین خاطر قبل از تخصیص، مقدار را کنترل کردهایم.
خروجی مثال فوق به شکل زیر است:
Mashed Potato-Tasty!
Mashed Potato-Not nice :(
Mashed Potato-Pretty good
Crispy Duck-Too hard
Crispy Duck-Loved it
Sachertorte-n/a
همانطور که مشاهده میکنید، خروجی به صورت غیر سلسله مراتبی و flat میباشد. از طریق دستور select و جایگزین کردن مقدار N/A به جای null، عنصر Sachertorte را در خروجی مهیا کردهایم .
یک راه جایگزین برای کنترل مقدار null در دستور select بازسازی متد DefaultIfEmpty بهصورتی است که به جای ایجاد مقدار null، یک نمونهی جدید از مدل review را با مقدار پیش فرض N/A ایجاد کند.
مثال:
var query = from recipe in recipes
join review in reviews on recipe.Id equals review.RecipeId
into reviewGroup
from rg in reviewGroup.DefaultIfEmpty(new Review { ReviewText = "N/A" })
select new //anonymous type
{
RecipeName = recipe.Name,
RecipeReview = rg.ReviewText
};
foreach (var item in query)
{
Console.WriteLine($"{item.RecipeName}-{item.RecipeReview}");
}
خروجی مثال بالا :
Mashed Potato-Tasty!
Mashed Potato-Not nice :(
Mashed Potato-Pretty good
Crispy Duck-Too hard
Crispy Duck-Loved it
Sachertorte-N/A