Povezovanje prek tabele povezav v PHP
Recimo, da je bil uporabnik v različnih mestih. V tem primeru bi lahko tabela z uporabniki imela naslednjo obliko:
| id | name | city |
|---|---|---|
| 1 | user1 | city1, city2, city3 |
| 2 | user2 | city1, city2 |
| 3 | user3 | city2, city3 |
| 4 | user4 | city1 |
Jasno je, da tako shranjevanje podatkov ni pravilno - mesta je treba prenesti v ločeno tabelo. Tukaj je:
| id | name |
|---|---|
| 1 | city1 |
| 2 | city2 |
| 3 | city3 |
Vendar moramo poskrbeti, da bo vsak uporabnik lahko skliceval na več mest. Z dvema tabelama tega ni mogoče storiti.
Potrebovali bomo tako imenovano tabelo povezav, ki bo povezovala uporabnika z njegovimi mesti.
V vsakem zapisu te tabele bo shranjena povezava med uporabnikom in enim mestom. Pri tem bo za enega uporabnika v tej tabeli toliko zapisov, v kolikor mestih je bil.
Tukaj je naša tabela povezav:
| 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 |
Tabela z uporabniki bo shranjevala samo imena uporabnikov, brez povezav:
| id | name |
|---|---|
| 1 | user1 |
| 2 | user2 |
| 3 | user3 |
| 4 | user4 |
| 5 | user5 |
Poizvedbe
Izvedimo poizvedbo, s katero poberemo uporabnike skupaj z njihovimi mesti. Za to bomo potrebovali narediti dva združevanja (join): prvo združevanje bo k uporabnikom pridružilo tabelo povezav, drugo združevanje pa bo po povezavah pridružilo mesta:
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
Rezultat poizvedbe
Rezultat naše poizvedbe v PHP bo vseboval ime vsakega uporabnika tolikokrat, s kolikor mesti je povezan:
<?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'],
];
?>
Bolj priročno bi bilo takšen seznam pretvoriti in ga spremeniti v naslednjega:
<?php
$res = [
['user1' => ['city1', 'city2', 'city3']],
['user2' => ['city1', 'city2']],
['user3' => ['city2', 'city3']],
['user4' => ['city1']],
];
?>
Napišimo kodo, ki izvede takšno pretvorbo:
<?php
$res = [];
foreach ($data as $elem) {
$res[$elem['user_name']][] = $elem['city_name'];
}
var_dump($res);
?>
Praktične naloge
Recimo, da izdelek lahko pripada več kategorijam. Opišite strukturo shranjevanja.
Napišite poizvedbo, ki bo poiskala izdelke skupaj z njihovimi kategorijami.
Prikažite pridobljene podatke v obliki seznama
ul tako, da v vsaki li najprej
stoji ime izdelka, za dvopičjem pa so ločene z vejico
kategorije tega izdelka.
Približno takole:
<ul>
<li>product1: category1, category2, category3</li>
<li>product2: category1, category3</li>
<li>product3: category1</li>
</ul>