选择列表

如前面的小节说明的那样,在SELECT命令中的表表达式通过组合表、 视图、删除行、分组等构造了一个中介性的虚拟表。这个表最后传递给 选择列表 处理。 选择列表判断最终实际输出虚拟表的哪些

选择列表项

最简单的选择列表是*,它输出表表达式生成的所有列。 否则,一个选择列表是一个逗号分隔的值表达式的列表(和在 值表达式 里定义的一样)。比如,它可能是一个列名列表:

SELECT a, b, c FROM ...

列名a, b,c要么是在FROM 子句里引用的表中列的实际名字,要么是 表和列别名 里解释的别名。选择列表中的名字空间和WHERE子句中的名字空间是一样的, 除非你使用了分组(这样的情况下它和HAVING子句中的名字空间也一样)。

如果多个表有重复的列名,那么你还必须给出表名字,例如:

SELECT tbl1.a, tbl2.a, tbl1.b FROM ...

当使用多个表时,给出表名还有助于引用该表的所有列:

SELECT tbl1.*, tbl2.a FROM ...

又见 WHERE子句

如果将值表达式用于选择列表,那么它在概念上向返回的表中增加了一个新的虚拟列。值表达式为结果中的每一行进行一次计算,用该行的数值替换任何表达式里引用的列。 不过选择列表中的这个表达式并非一定要引用来自FROM子句中表表达式里面的列, 比如,它也可以是任意常量算术表达式。

列标签

选择列表中的列表项可以赋予名字,以便于进一步的处理。这里进一步的处理是一个可选的查询声明和用户应用(比如,用于展示的列头)。例如:

SELECT a AS value, b + c AS sum FROM ...

如果没有使用AS声明列名字,那么系统将赋予一个缺省值。 对于简单的列引用,它是该列的名字。对于函数调用,它是该函数的名字。 对于复杂表达式,系统会生成一个通用的名字。

Note

输出列的命名和在FROM子句里的命名是不一样的(参阅 表和列别名 )。 这样就允许你对同一个列命名两次, 而选择列表中新取的名字将被最终输出。

DISTINCT

在处理完选择列表之后,生成的表可以选择删除重复行。直接在SELECT 后面写上DISTINCT关键字即可:

SELECT DISTINCT select_list ...

如果不用DISTINCT你可以用ALL声明保留所有行的缺省行为

显然,如果两行里至少有一个列值不同,那么我们认为这两行是独立的。 NULL 在这里被认为是相同的。

另外,我们还可以用表达式来判断什么样的行可以认为是独立的:

SELECT DISTINCT ON (expression [, expression ...]) select_list ...

这里的expression是一个值表达式, 它为每一行计算。如果一组行计算出的该表达式的值都相同,那么就认为这些行是重复的, 并只输出第一行。请注意这里的”第一行”是不可预料的, 除非你在足够多的列上对该查询进行了排序,保证到达DISTINCT 过滤器时行的顺序是唯一的(DISTINCT ON将在ORDER BY排序之后处理)。

DISTINCT ON子句不是 SQL 标准的一部分,有时候被认为是一个糟糕的风格, 因为它的结果是不可判定的。如果用有可选的GROUP BY和在FROM 中的子查询可以达到目的,那么我们可以避免使用这个构造,但是通常它是更方便的方法。