การเชื่อมโยงผ่านตารางความสัมพันธ์ใน PHP
สมมติว่าตอนนี้ผู้ใช้เคยอยู่ในเมืองต่างกันหลายเมือง ในกรณีนี้ตารางผู้ใช้อาจจะมีลักษณะดังต่อไปนี้:
| id | name | city |
|---|---|---|
| 1 | user1 | city1, city2, city3 |
| 2 | user2 | city1, city2 |
| 3 | user3 | city2, city3 |
| 4 | user4 | city1 |
เป็นที่เข้าใจว่าการเก็บข้อมูลแบบนี้ไม่ถูกต้อง - จำเป็นต้องแยกเมืองออกมาเก็บในตารางแยกต่างหาก นี่คือตารางนั้น:
| id | name |
|---|---|
| 1 | city1 |
| 2 | city2 |
| 3 | city3 |
อย่างไรก็ตาม เราต้องทำให้ผู้ใช้แต่ละคนสามารถอ้างอิงไปยังเมืองได้หลายเมือง การใช้แค่สองตารางทำให้ทำสิ่งนี้ไม่ได้
เราจำเป็นต้องนำสิ่งที่เรียกว่า ตารางความสัมพันธ์ มาใช้ ซึ่งจะทำหน้าที่เชื่อมโยงผู้ใช้กับเมืองของเขา
ในแต่ละเรคคอร์ดของตารางนี้จะเก็บความสัมพันธ์ระหว่างผู้ใช้หนึ่งคนกับเมืองหนึ่งเมือง ในขณะเดียวกัน สำหรับผู้ใช้หนึ่งคนในตารางนี้จะมีเรคคอร์ดจำนวนเท่ากับจำนวนเมืองที่เขาเคยไป
นี่คือตารางความสัมพันธ์ของเรา:
| id | user_id | city_id |
|---|---|---|
| 1 | 1 | 1 |
| 2 | 1 | 2 |
| 3 | 1 | 3 |
| 4 | 2 | 1 |
| 5 | 2 | 2 |
| 6 | 3 | 2 |
| 7 | 3 | 3 |
| 8 | 4 | 1 |
ตารางผู้ใช้จะเก็บเพียงชื่อผู้ใช้เท่านั้น โดยไม่มีการเชื่อมโยง:
| id | name |
|---|---|
| 1 | user1 |
| 2 | user2 |
| 3 | user3 |
| 4 | user4 |
| 5 | user5 |
การสอบถามข้อมูล
ลองมาสร้างคำสั่งสอบถามที่ใช้ดึงข้อมูลผู้ใช้พร้อมกับเมืองของพวกเขากัน ในการนี้เราจะต้องใช้การ join สองครั้ง: การ join ครั้งแรกจะเชื่อมตารางผู้ใช้กับตารางความสัมพันธ์ และการ join ครั้งที่สองจะเชื่อมเมืองผ่านความสัมพันธ์:
SELECT
users.name as user_name, cities.name as city_name
FROM
users
LEFT JOIN users_cities ON users_cities.user_id=users.id
LEFT JOIN cities ON users_cities.city_id=cities.id
ผลลัพธ์ของการสอบถาม
ผลลัพธ์จากการสอบถามของเราใน PHP จะประกอบด้วยชื่อของผู้ใช้แต่ละคนซ้ำจำนวนครั้งเท่ากับจำนวนเมืองที่เขาเกี่ยวข้อง:
<?php
$arr = [
['user_name' => 'user1', 'city_name' => 'city1'],
['user_name' => 'user1', 'city_name' => 'city2'],
['user_name' => 'user1', 'city_name' => 'city3'],
['user_name' => 'user2', 'city_name' => 'city1'],
['user_name' => 'user2', 'city_name' => 'city2'],
['user_name' => 'user3', 'city_name' => 'city2'],
['user_name' => 'user3', 'city_name' => 'city3'],
['user_name' => 'user4', 'city_name' => 'city1'],
];
?>
มันจะสะดวกกว่าถ้าเราสามารถแปลงอาร์เรย์แบบนี้และเปลี่ยนมันเป็นรูปแบบต่อไปนี้:
<?php
$res = [
['user1' => ['city1', 'city2', 'city3']],
['user2' => ['city1', 'city2']],
['user3' => ['city2', 'city3']],
['user4' => ['city1']],
];
?>
มาเขียนโค้ดที่ทำการแปลงแบบนี้กัน:
<?php
$res = [];
foreach ($data as $elem) {
$res[$elem['user_name']][] = $elem['city_name'];
}
var_dump($res);
?>
แบบฝึกหัดภาคปฏิบัติ
สมมติว่าสินค้าสามารถอยู่ในหลายหมวดหมู่ อธิบายโครงสร้างการจัดเก็บข้อมูล
เขียนคำสั่งสอบถามที่ดึงข้อมูลสินค้าพร้อมกับหมวดหมู่ของสินค้านั้น
แสดงผลข้อมูลที่ได้ในรูปแบบรายการ ul โดยที่ในแต่ละ li จะมีชื่อสินค้าอยู่ตอนต้น ตามด้วยเครื่องหมายโคลอนและชื่อหมวดหมู่ของสินค้านั้นคั่นด้วยเครื่องหมายจุลภาค ตัวอย่างเช่น:
<ul>
<li>product1: category1, category2, category3</li>
<li>product2: category1, category3</li>
<li>product3: category1</li>
</ul>