6

SQL/dbms を学習し、Postgres を使用しています。特定の列に特定の値を持つ行を返したい。たとえば、テーブルCarpetsとで、色が である行Curtainsの s を取得したいとします。これにはJOINが必要だと思いますが、どのタイプかわかりません。id'light yellow'

これが私が持っているものです:

SELECT id
  FROM Carpets
  WHERE colour = 'light yellow'
        INNER JOIN Curtains ON Carpets.colour = Curtains.colour;

両方のテーブルにid属性があります。

学習についてJOIN、どちらを先に学習すればよいですか?それらすべてを一度に学習しようとすると、自分の足を撃ちます(さまざまなリソースにはさまざまな「バリアント」が含まれているため).

重要idカーテンとカーペットの両方が「明るい黄色」の場合にのみ返品されるという回答を探していました。

4

4 に答える 4

13

この特定の質問に関する Meta での質問を読んだ後、3 つの回答すべてが実際に正しい理由を説明させてください。

3 つの回答すべての例と、彼らが取り組んでいるスキーマを含めました。

Database changed
mysql> create table carpet(id int(3), material varchar(10), color varchar(15));
Query OK, 0 rows affected (0.02 sec)

mysql> create table curtain(id int(3), material varchar(10), color varchar(15));
Query OK, 0 rows affected (0.00 sec)

(挿入ステートメントの束)

mysql> select * from carpet;
+------+-----------+--------------+
| id   | material  | color        |
+------+-----------+--------------+
|    1 | wool      | Light Yellow |
|    2 | wool      | Beige        |
|    3 | polyester | Light Yellow |
|    4 | polyester | Light Red    |
+------+-----------+--------------+
4 rows in set (0.00 sec)

mysql> select * from curtain;
+------+----------+--------------+
| id   | material | color        |
+------+----------+--------------+
|    1 | Velvet   | Purple       |
|    2 | cotton   | White        |
|    3 | cotton   | Light Yellow |
|    4 | cotton   | Light Blue   |
+------+----------+--------------+
4 rows in set (0.00 sec)

交差は 2 つの選択ステートメントを使用し、一致する結果を返します。この場合、「Light Yellow」の色が一致するすべての行を探しています。

MySQL はサポートされていないため、MySQL での例を示すことはできません (以下に示すように、同じ結果を得る必要はありません)。

それぞれが 'Light Yellow' の色のみを許可する where 句を持つ 2 つの select ステートメントのユニオン クエリは、同じデータを返します。ユニオンを使用して一致しないデータを返すことはできますが、各 select ステートメントの where 句は、必要な行のみを実際に返すことを意味します。

mysql> select id, material, color from carpet
    -> union 
    -> select id, material, color from curtain;
+------+-----------+--------------+
| id   | material  | color        |
+------+-----------+--------------+
|    1 | wool      | Light Yellow |
|    2 | wool      | Beige        |
|    3 | polyester | Light Yellow |
|    4 | polyester | Light Red    |
|    1 | Velvet    | Purple       |
|    2 | cotton    | White        |
|    3 | cotton    | Light Yellow |
|    4 | cotton    | Light Blue   |
+------+-----------+--------------+
8 rows in set (0.00 sec)

うわぁ、まずいですよね?もちろん、where 句は指定しませんでした。

mysql> select id, material, color from carpet where color='Light Yellow'
    -> union
    -> select id, material, color from curtain where color='Light Yellow';
+------+-----------+--------------+
| id   | material  | color        |
+------+-----------+--------------+
|    1 | wool      | Light Yellow |
|    3 | polyester | Light Yellow |
|    3 | cotton    | Light Yellow |
+------+-----------+--------------+
3 rows in set (0.00 sec)

色で 2 つのテーブルを結合すると、両方のテーブルの行を 1 行のデータとして返すことができます。項目の色に 2 つのテーブルの結合を指定し、where 句を使用して探している行のみを返すことができます。

mysql> select a.id, a.material, a.color, b.id, b.material 
    -> from curtain a join carpet b on a.color=b.color;
+------+----------+--------------+------+-----------+
| id   | material | color        | id   | material  |
+------+----------+--------------+------+-----------+
|    3 | cotton   | Light Yellow |    1 | wool      |
|    3 | cotton   | Light Yellow |    3 | polyester |
+------+----------+--------------+------+-----------+
2 rows in set (0.00 sec)

ご覧のとおり、これは色が一致する行のみを返し、結果セットの 1 つの行に両方のテーブルの列を含めることができました。

ここで、両方のテーブルに「ライト イエロー」以外に一致する結果がないため、明らかにこれをうまく計画していませんでした。そのため、さらにいくつかのエントリを追加すると、次のようになります。

mysql> select * from curtain;
+------+----------+--------------+
| id   | material | color        |
+------+----------+--------------+
|    1 | Velvet   | Purple       |
|    2 | cotton   | White        |
|    3 | cotton   | Light Yellow |
|    4 | cotton   | Light Blue   |
|    5 | Wool     | White        |
|    6 | Fluff    | Beige        |
+------+----------+--------------+
6 rows in set (0.00 sec)

mysql> select * from carpet;
+------+-----------+--------------+
| id   | material  | color        |
+------+-----------+--------------+
|    1 | wool      | Light Yellow |
|    2 | wool      | Beige        |
|    3 | polyester | Light Yellow |
|    4 | polyester | Light Red    |
|    5 | Fluff     | Light Blue   |
+------+-----------+--------------+
5 rows in set (0.00 sec)

これをもう一度実行すると、次のようになります。

mysql> select a.id, a.material, a.color, b.id, b.material 
    -> from curtain a join carpet b on a.color=b.color;
+------+----------+--------------+------+-----------+
| id   | material | color        | id   | material  |
+------+----------+--------------+------+-----------+
|    3 | cotton   | Light Yellow |    1 | wool      |
|    3 | cotton   | Light Yellow |    3 | polyester |
|    4 | cotton   | Light Blue   |    5 | Fluff     |
|    6 | Fluff    | Beige        |    2 | wool      |
+------+----------+--------------+------+-----------+
4 rows in set (0.00 sec)

いやいや!

ここで、join 句と where 句を一緒に使用します。

mysql> select a.id, a.material, a.color, b.id, b.material 
    -> from curtain a join carpet b on a.color=b.color 
    -> where a.color='Light Yellow';
+------+----------+--------------+------+-----------+
| id   | material | color        | id   | material  |
+------+----------+--------------+------+-----------+
|    3 | cotton   | Light Yellow |    1 | wool      |
|    3 | cotton   | Light Yellow |    3 | polyester |
+------+----------+--------------+------+-----------+
2 rows in set (0.00 sec)

おわかりのように、SQL では、テーブル内に同じデータのバリエーションが存在するよりも、さまざまな手段で同じ結果を得る方法の方が多いことがよくあります。

編集:わかりました。すべてのデータが一致する行のみが必要な場合は、結合構文に含めるだけです。

mysql> select a.id, a.material, a.color, b.id, b.material 
    -> from curtain a 
    -> join carpet b on a.color=b.color
    -> and a.id=b.id
    -> where a.color='Light Yellow';
+------+----------+--------------+------+-----------+
| id   | material | color        | id   | material  |
+------+----------+--------------+------+-----------+
|    3 | cotton   | Light Yellow |    3 | polyester |
+------+----------+--------------+------+-----------+
1 row in set (0.00 sec)

ご覧のとおり、ここで、idcolorフィールドの両方が 2 つのテーブル間で一致する必要があることを結合に伝えます。結果がそれを物語っています。さて、この場合、材料が異なるため、技術的にはすべての列を一致させることはできませんでした。さらに一致させたい場合、ID、マテリアル、および色が一致する一致するレコードがないため、クエリは結果を返しませんが、構文は次のようになります。

mysql> select a.id, a.material, a.color, b.id, b.material 
    -> from curtain a 
    -> join carpet b on a.color=b.color
    -> and a.id=b.id
    -> and a.material=b.material
    -> where a.color='Light Yellow';
Empty set (0.00 sec)

ただし、ほとんどの場合、すべての列が一致することは望ましくありません。多くの場合、テーブルには、そのテーブルでのみ使用される ID があり、値は自動的に増分されます。そのテーブル内の一意の行を識別するために使用したいが、関連のないテーブルを照合するために使用したくない。どちらかといえば、素材と色を一致させることをお勧めしますが、IDは除外してください.

于 2012-09-17T08:22:58.467 に答える
2

結果は 2 つの互いに素なテーブルの結果であるため、実際には代わりにユニオンが必要です。

select id, 'curtains' as source from curtains where color = 'lightyellow'
  union 
select id, 'carpets' as source from carpets where color = 'lightyellow'

結合については、まとめて学習してください。それらは互いにわずかに異なるだけです。

于 2012-09-16T02:14:27.767 に答える
2

2 つのテーブルでのマッチングの結果が必要な場合は、次のようにしてください。

select id
  from curtains
      ,carpets
  where curtain.color = carpets.color;

これはidを返しますcurtain.color = carpets.color

于 2012-09-16T02:52:48.540 に答える
2

これらの両方のクエリにより、必要な結果が得られます....

SELECT Carperts.id
  FROM Carpets INNER JOIN Curtains ON Carpets.colour = Curtains.colour
  and colour = 'light yellow';


SELECT Carperts.id
  FROM Carpets INNER JOIN Curtains ON Carpets.colour = Curtains.colour
  WHERE colour = 'light yellow';
于 2012-09-16T09:59:36.853 に答える