Свързване чрез таблица за свързване в 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 |
Заявки
Нека направим заявка, с която да извлечем потребителите заедно с техните градове. За това ще ни трябват две съединения: първото съединение ще присъедини към потребителите таблицата за свързване, а второто съединение по връзките ще присъедини градовете:
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>