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 |
しかし、各ユーザーが複数の都市を参照できるようにする必要があります。2つのテーブルだけではこれは不可能です。
ここで、いわゆる 関連テーブル を導入する必要があります。これはユーザーとその都市を関連付けます。
このテーブルの各レコードには、ユーザーと1つの都市との関連が保存されます。このテーブルには、ユーザーが訪れた都市の数だけ、そのユーザーのレコードが存在します。
これが関連テーブルです:
| 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 |
クエリ
ユーザーとその都市を一緒に取得するクエリを作成してみましょう。そのためには、2つの結合(JOIN)が必要です。最初の結合でユーザーに関連テーブルを結合し、2番目の結合で関連を通じて都市テーブルを結合します:
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 リストとして表示してください。例は以下の通りです:
<ul>
<li>product1: category1, category2, category3</li>
<li>product2: category1, category3</li>
<li>product3: category1</li>
</ul>