Звязванне праз табліцу звязку ў Python
Хай цяпер юзер быў у розных гарадах. У гэтым выпадку табліца з юзерамі магла б мець наступны выгляд:
| 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
Вынік запыту
Вынік нашага запыту ў Python будзе ўтрымліваць імя кожнага юзера столькі разоў, з колькімі гарадамі ён звязаны:
{'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}
Зручней было б пераканвертаваць такі слоўнік і ператварыць яго ў наступны:
{
'user1': ['city1', 'city2', 'city3'],
'user2': ['city1', 'city2'],
'user3': ['city2', 'city3'],
'user4': ['city1'],
'user5': []
}
Напішам код, які выконвае такую канвертацыю.
Зробім пусты слоўнік user_cities_dct,
у які будзем паступова ўводзіць даныя
пра юзераў і гарады, у якіх яны пабывалі.
У цыкле for аб'яўляем дзве зменныя
user_name і city_name, у
якіх будуць захоўвацца імя юзера і назва
горада. Далей прапісваем умову - калі
юзера няма ў user_cities_dct, то ён
дадасца ў гэты слоўнік у якасці ключа.
Таксама ўкажам умову, што калі city_name
не None, то ён дадасца як значэнне
ключа:
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)
Практычныя задачы
Хай тавар можа належаць некалькім катэгорыям. Распішыце структуру заховання.
Напішыце запыт, які дастане тавары разам з іх катэгорыямі.
Выведзіце атрыманыя даныя ў выглядзе спісу
ul так, каб у кожнай li
ўпачатку стаяла імя прадукту, а пасля
двукроп'я праз коску пералічваліся
катэгорыі гэтага прадукту. Прыблізна так:
<ul>
<li>product1: category1, category2, category3</li>
<li>product2: category1, category3</li>
<li>product3: category1</li>
</ul>