Prepojenie cez tabuľku vzťahov v Pythone
Predpokladajme, že používateľ bol v rôznych mestách. V tomto prípade by tabuľka s používateľmi mohla mať nasledujúci tvar:
| id | name | city |
|---|---|---|
| 1 | user1 | city1, city2, city3 |
| 2 | user2 | city1, city2 |
| 3 | user3 | city2, city3 |
| 4 | user4 | city1 |
Je zrejmé, že takto ukladať údaje nie je správne - mestá je potrebné vyčleniť do samostatnej tabuľky. Tu je:
| id | name |
|---|---|
| 1 | city1 |
| 2 | city2 |
| 3 | city3 |
Avšak potrebujeme zabezpečiť, aby každý používateľ mohol odkazovať na viacero miest. Pomocou dvoch tabuliek to nie je možné.
Budeme potrebovať zaviesť takzvanú tabuľku vzťahov, ktorá bude spájať používateľa s jeho mestami.
V každom zázname tejto tabuľky bude uložený vzťah medzi používateľom a jedným mestom. Pri tom pre jedného používateľa bude v tejto tabuľke tolko záznamov, v koľkých mestách bol.
Tu je naša tabuľka vzťahov:
| 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 |
Tabuľka s používateľmi bude ukladať iba mená používateľov, bez vzťahov:
| id | name |
|---|---|
| 1 | user1 |
| 2 | user2 |
| 3 | user3 |
| 4 | user4 |
| 5 | user5 |
Dopytovanie
Vytvorme dopyt, pomocou ktorého získame používateľov spolu s ich mestami. Pre to budeme potrebovať vykonať dve spojenia: prvé spojenie pripojí k používateľom tabuľku vzťahov, a druhé spojenie podľa vzťahov pripojí mestá:
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
Výsledok dopytu
Výsledok nášho dopytu v Pythone bude obsahovať meno každého používateľa toľkokrát, s koľkými mestami je spojený:
{'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'}
{'user_name': 'user5', 'city_name': None}
Bolo by pohodlnejšie takýto slovník prekonvertovať a premeniť ho na nasledujúci:
{
'user1': ['city1', 'city2', 'city3'],
'user2': ['city1', 'city2'],
'user3': ['city2', 'city3'],
'user4': ['city1'],
'user5': []
}
Napíšme kód, ktorý vykoná takúto konverziu.
Vytvorme prázdny slovník user_cities_dct,
do ktorého budeme postupne vkladať údaje
o používateľoch a mestách, v ktorých boli.
V cykle for deklarujeme dve premenné
user_name a city_name, v
ktorých budú uložené meno používateľa a názov
mesta. Ďalej zadáme podmienku - ak
používateľ nie je v user_cities_dct, tak sa
pridá do tohto slovníka ako kľúč.
Tiež uvedieme podmienku, že ak city_name
nie je None, tak sa pridá ako hodnota
kľúča:
result = cursor.fetchall()
user_cities_dct = {}
for row in result:
user_name = row['user_name']
city_name = row['city_name']
if user_name not in user_cities_dct:
user_cities_dct[user_name] = []
if city_name is not None:
user_cities_dct[user_name].append(city_name)
print(user_cities_dct)
Praktické úlohy
Predpokladajme, že produkt môže patriť do viacerých kategórií. Popíšte štruktúru uloženia.
Napíšte dopyt, ktorý získa produkty spolu s ich kategóriami.
Zobrazte získané údaje vo forme zoznamu
ul tak, aby v každom li
na začiatku stálo meno produktu, a za
dvojbodkou oddelené čiarkou boli vymenované
kategórie tohto produktu. Približne takto:
<ul>
<li>product1: category1, category2, category3</li>
<li>product2: category1, category3</li>
<li>product3: category1</li>
</ul>