حسام جعفرزاده
حسام جعفرزاده
خواندن ۴ دقیقه·۳ سال پیش

آموزش postgres - بررسی self-join


در این مطلب در مورد این یاد می‌گیریم که چجوری با استفاده از self-join یک جدول رو با خودش مقایسه کنیم.

قبل از هر چیز توصیه میکنیم که اگر با joinها آشنایی ندارید، نخست آموزش انواع join رو مطالعه بفرمایید.

در واقع self-join یک join معمولی است که یک جدول را با خودش join می‌کند. بیشتر اوقات از این join برای اجرای کوئری‌هایی استفاده می‌کنیم که نتایج حالت خوشه‌ای و یا ارث‌بری دارند و یا حتی زمان‌هایی که میخواهیم اطلاعات یک سطر از یک جدول رو با بقیه‌ی سطر‌های همون جدول مقایسه کنیم.

در self-join تمام ارجاعات ما به یک جدول است اما برای ایجاد تفاوت باید هر بار برای جدول اسمی جدید در نظر بگیریم (table aliases)، به نحوه‌ی نوشتن توجه کنید:

SELECT select_list FROM table_name t1 INNER JOIN table_name t2 ON join_predicate;

در دستور بالا جدول table_name را، که موقتا آنرا t1 نام گذاری کردیم، به همان جدول table_name که این‌بار موقتا آنرا t2 نام گذاری کردیم inner join زدیم. توجه کنید که ما می‌توانیم دستور بالا را به کمک left join و right join هم بنویسیم.

SELECT select_list FROM table_name t1 LEFT JOIN table_name t2 ON join_predicate;



اجازه بدید که آموزش رو با مثال پیش ببریم،

قصد داریم تا یک ساختار خوشه‌ای رو با استفاده از self-join پیاده سازی کنیم. فرض کنید که ساختار زیر رو برای یک سازمان داریم:

self-join
self-join


برای پیاده سازی این دستور ابتدا یک جدول employee می‌سازیم و داده‌های بالا را در آن ذخیره می‌کنیم.

CREATE TABLE employee ( employee_id INT PRIMARY KEY, first_name VARCHAR (255) NOT NULL, last_name VARCHAR (255) NOT NULL, manager_id INT, FOREIGN KEY (manager_id) REFERENCES employee (employee_id) ON DELETE CASCADE ); INSERT INTO employee ( employee_id, first_name, last_name, manager_id ) VALUES (1, 'Windy', 'Hays', NULL), (2, 'Ava', 'Christensen', 1), (3, 'Hassan', 'Conner', 1), (4, 'Anna', 'Reeves', 2), (5, 'Sau', 'Norman', 2), (6, 'Kelsie', 'Hays', 3), (7, 'Tory', 'Goff', 3), (8, 'Salley', 'Lester', 3);


توجه کنید که در جدول بالا ستون manager_id در واقع اشاره می‌کنید به ستون employee_id از همین جدول و به عبارت ساده بیان کننده این است که، مدیرِ شخصِ Salley، کارمندِ شماره‌ی ۳ است که Hassan نام دارد. زمانی که ستون manager_id خالی باشد به این معنا است که این شخص به خودش مقام بالا است و کسی manager او نیست.

حال قصد داریم با یک query مشخص کنیم که در داده‌های این جدول چه کسی مدیر فرد دیگری است:

SELECT e.first_name || ' ' || e.last_name employee, m.first_name || ' ' || m.last_name manager FROM employee e INNER JOIN employee m ON m.employee_id = e.manager_id ORDER BY manager;

در دستور بالا دو بار به جدول employee ارجاع دادیم که با استفاده از table aliases (تعیین نام موقت برای جدول) یک بار آنرا جدول e و یک بار دیگر آنرا جدول m نام گذاری کردیم. در دستور select هم از تعیین نام موقت یا column aliases استفاده کردیم. (برای کسب اطلاعات بیشتر در این خصوص،‌ آموزش نام گذاری موقت در دیتابیس رو ببینید).

در join بالا تعیین کردیم که داده‌های ستون employee_id از جدول m رو با داده‌های ستون manager_id از جدول e مقایسه شوند و در صورتی که برابر باشند در نتایج ما قرار گیرند.

به خروجی دستور توجه کنید:

self join
self join

اگر توجه کنید متوجه می‌شوید که top manager یا افرادی که manager_id آنها null هست در این لیست وجود ندارند.

برای حل این مسئله نیز می‌تونیم کوئری رو با استفاده از LEFT JOIN به شکل زیر تغییر بدیم.

SELECT e.first_name || ' ' || e.last_name employee, m .first_name || ' ' || m .last_name manager FROM employee e LEFT JOIN employee m ON m .employee_id = e.manager_id ORDER BY manager;

خروجی دستور بالا به این شکل خواهد بود:

self join
self join



بپردازیم به یک مثال دیگر که در آن قصد داریم سطر‌های یک جدول رو با سطر‌های دیگر از همون جدول مقایسه کنیم:

جدول زیر رو در نظر بگیرید که برای نگهداری اطلاعات فیلم‌ها ازش استفاده می‌کنیم.

film table
film table

در جدول بالا یک ستون به اسم length داریم که مدت زمان فیلم رو توی خودش نگه میداره و ما قصد داریم فیلم‌هایی که مدت زمان یکسانی رو دارند رو لیست کنیم. برای این کار از دستور زیر کمک می‌گیریم:

SELECT f1.title, f2.title, f1.length FROM film f1 INNER JOIN film f2 ON f1.film_id <> f2.film_id AND f1.length = f2.length;

در دستور بالا جدول film رو با خودش inner join زدیم و شرط گذاشتیم که فیلم‌هایی که طول یکسانی دارند لیست شوند در با f1.film_id <> f2.film_id مطمئن شدیم که یک فیلم با خودش مقایسه نشود.

خروجی دستور بالا به شکل زیر خواهد بود:

self join
self join


امیدوارم مطالب براتون مفید واقع شده باشه و لایک و کامنت رو فراموش نکنید.

پایان/

اتصال در دیتابیساموزش دیتابیساموزش پستگرسبانک اطلاعاتیpostgres
به دنبال یادگیری بیشتر - اطلاعات بیشتر در مورد من در: hesij.ir
شاید از این پست‌ها خوشتان بیاید