如果条件在两个表的行中匹配,则创建返回id的查询

时间:2012-09-16 02:01:56

标签: sql postgresql

我正在学习SQL / dbms并使用Postgres。我想返回在某列中都具有特定值的行。例如,在表格CarpetsCurtains中,我希望得到颜色为id的{​​{1}}行。我想我需要JOIN,但我不确定是什么类型。

这就是我所拥有的:

'light yellow'

两个表都有SELECT id FROM Carpets WHERE colour = 'light yellow' INNER JOIN Curtains ON Carpets.colour = Curtains.colour; 属性。

关于学习id,我应该先学习哪一门?如果我尝试一次性学习所有这些,我就会在脚下射击自己(因为不同的资源包括不同的'变种')。

重要我正在寻找一个答案,只有当窗帘和地毯都是“浅黄色”时才会返回JOIN

4 个答案:

答案 0 :(得分:13)

在Meta上阅读了关于这个特定问题的your question,让我解释为什么所有三个答案都是正确的 - 就像你解决这个问题一样。

我已经包含了所有三个答案的示例以及他们正在处理的架构:

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)

intersect使用两个select语句并返回匹配结果。在这种情况下,您要查找具有匹配颜色“浅黄色”的所有行。

我不能在MySQL中给你一个例子,因为它不支持它(如下所示,不需要给出相同的结果)。

两个select语句的联合查询,每个语句都带有一个只允许“浅黄色”颜色的where子句,它将返回相同的数据。虽然联合可用于返回不匹配的数据,但每个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)

颜色上两个表之间的连接将允许您从单个数据行中的两个表中返回行。您可以在两个表上为项颜色指定连接,并使用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)

正如您所看到的,这只返回了具有匹配颜色的行,并允许您在结果集的单行中包含两个表中的列。

现在,我显然没有很好地计划这个,因为除了两个表中的'浅黄色'之外我没有其他匹配的结果,所以如果我添加更多的条目,我们得到这个:

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字段必须在两个表之间匹配 - 并且结果不言而喻。现在,在这种情况下,我在技术上仍然与所有列都不匹配,因为材料不同。如果你想进一步匹配,查询将不会返回任何结果,因为我没有匹配的记录,其中id,material和color匹配,但语法如下:

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)

但是,在大多数情况下,您不希望 all 列匹配。通常,表具有仅用于该表的ID,并且是自动递增的值。您希望使用它来标识 表中的唯一行,但不要使用它来匹配不相关的表。如果有的话,我会建议你在材料和颜色上进行匹配 - 但是不要使用它。

答案 1 :(得分:2)

由于结果是针对两个不相交的表,实际上你需要一个联合:

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

关于联接,一起学习它们。它们只是彼此的微小变化。

答案 2 :(得分:2)

如果你想在两个表中匹配结果,那么试试这个:

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

这将返回curtain.color = carpets.color

的ID

答案 3 :(得分: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';