#SELECT

# 通用描述

SELECT用于执行对数据库的表、视图、AC等的查询操作。

YashanDB支持丰富多样的查询操作,包括但不限于:

其中,对AC可以执行上述列示的多表连接、排序、分组等基本查询。

在分布式部署中,全局动态视图相关的操作均在各节点本地执行,最后再汇总至CN。

# 语句定义

select::=

syntax
subquery for_update_clause

subquery::=

syntax
query_block subquery set_oper subquery ( subquery ) order_by_clause row_limiting_clause offset_fetch_clause

query_block::=

syntax
with_clause SELECT hint DISTINCT select_list FROM table_reference join_clause ( join_clause ) where_clause hierarchical_query_clause group_by_clause

with_clause::=

syntax
WITH cte_clause ,

cte_clause::=

syntax
cte_name ( column_alias , ) AS ( subquery )

select_list::=

syntax
* expr_clause AS alias ,

expr_clause::=

syntax
( query_block expr case_clause udt_expr )

case_clause::=

syntax
CASE simple_case_expression searched_case_expression else_clause END

simple_case_expression::=

syntax
expr WHEN comparison_expr THEN return_expr

searched_case_expression::=

syntax
WHEN condition THEN return_expr

else_clause::=

syntax
ELSE else_expr

table_reference::=

syntax
query_name subquery sample_clause flashback_query_clause t_alias pivot_clause ,

query_name::=

syntax
schema . table_name dblink synonym_name partition_extension_clause SLICE ( slice_id ) view_name table_collection_expression

partition_extension_clause::=

syntax
PARTITION ( partition ) FOR ( partition_key_value , ) SUBPARTITION ( subpartition ) FOR ( subpartition_key_value , )

table_collection_expression::=

syntax
TABLE ( collection_expression )

sample_clause::=

syntax
SAMPLE ( sample_percent ) SEED ( seed_value )

flashback_query_clause::=

syntax
AS OF SCN TIMESTAMP expr

pivot_clause::=

syntax
PIVOT ( aggregate_expression AS alias , aggregate_expression AS alias FOR column_expression IN ( const_expression AS alias , const_expression AS alias ) )

join_clause::=

syntax
table_reference inner_cross_join_clause outer_join_clause

inner_cross_join_clause::=

syntax
INNER JOIN table_reference ON condition CROSS JOIN table_reference

outer_join_clause::=

syntax
outer_join_type JOIN table_reference ON condition

outer_join_type::=

syntax
LEFT RIGHT FULL OUTER

where_clause::=

syntax
WHERE condition

hierarchical_query_clause::=

syntax
connect_by_clause start_with_clause order_siblings_by_clause start_with_clause connect_by_clause order_siblings_by_clause

connect_by_clause::=

syntax
CONNECT BY NOCYCLE condition

start_with_clause::=

syntax
START WITH condition

order_siblings_by_clause::=

syntax
ORDER SIBLINGS BY expr position c_alias ASC DESC NULLS FIRST NULLS LAST ,

group_by_clause::=

syntax
group_by_clause1 group_by_clause2

group_by_clause1::=

syntax
GROUP BY expr , rollup_cube_clause grouping_sets_clause HAVING condition

rollup_cube_clause::=

syntax
ROLLUP CUBE ( expr , )

grouping_sets_clause::=

syntax
GROUPING SETS ( expr , )

group_by_clause2::=

syntax
HAVING condition GROUP BY expr , rollup_cube_clause grouping_sets_clause

set_oper::=

syntax
UNION INTERSECT MINUS EXCEPT ALL

order_by_clause::=

syntax
ORDER BY expr position c_alias ASC DESC NULLS FIRST NULLS LAST ,

row_limiting_clause::=

syntax
LIMIT expr offset expr

offset_fetch_clause::=

syntax
OFFSET offset ROW ROWS FETCH FIRST NEXT rowcount ROW ROWS ONLY

for_update_clause::=

syntax
FOR UPDATE OF schema . table_name view_name synonym_name . t_alias column_name NOWAIT WAIT ntimes SKIP LOCKED

# subquery

YashanDB的SELECT语句支持嵌套查询(对封装在()中的SELECT语句的结果进行SELECT查询)和合并查询(用集合合并多个SELECT语句的查询结果),不同方式的查询可交叉及多层级使用,例如:

SELECT * FROM (SELECT a,b FROM (SELECT cl_a a,cl_b b FROM table_a WHERE ...
               					UNION ALL
               					SELECT cl_c a,cl_d b FROM table_b WHERE ...
                               ) 
               WHERE ...
               )
WHERE ...

每一个完整的SELECT查询语句(包括其下级)均可称为一个子查询(subQuery)。

对于子查询中的select_list列项:

  • 若这些列项显式指定了别名(例如为列c1_a定义别名a),系统以此别名作为子查询返回结果集的列名,YashanDB对显式指定别名的长度限制为64字节。

  • 若这些列项未显式指定别名,系统默认以该项列名作为子查询返回结果集的列名,且超过64字节截断,此时,如子查询select_list中存在名称前64字节相同的列,将导致返回结果集产生相同名称的列而触发系统报错。

示例

SELECT * FROM (SELECT
                 1111111111111111111111111111111111111111111111111111111111111111,
                 11111111111111111111111111111111111111111111111111111111111111112
               FROM dual);

[3:1]YAS-04301 ambiguous column

# query_block

含有单个SELECT的查询语句称为一个查询块(queryBlock)。

# with_clause

该语句用于定义CTE(Common Table Expression),在某个子查询被多次使用时,可以直接使用CTE名称而不需要重复写该SELECT语句。

# cte_clause

CTE定义,包括:

  • cte_name:名称,作为其他地方引用的标识。
  • column_alias:定义CTE的列项,其中列字段应该来自于其后的子查询中定义的列项。
  • subquery:用于创建CTE的子查询。

示例

--WITH语句用于获取area表中'01'和'04'地区的代码和名称,在后面的查询中对branches表获取关联地区名称数据时,不需要多次写该语句,而只用和q_area做关联查询即可
WITH q_area(ano,aname) AS (SELECT area_no ano,area_name aname FROM area WHERE area_no IN ('01','04'))
SELECT bno,bname,aname FROM (
SELECT a.branch_no bno,a.branch_name bname,b.aname aname FROM branches a,q_area b WHERE a.area_no=b.ano AND a.area_no='01'
UNION
SELECT a.branch_no bno,a.branch_name bname,b.aname aname FROM branches a,q_area b WHERE a.area_no=b.ano AND a.area_no='04'
);
BNO   BNAME             ANAME       
----- ----------------- -------------
0101  上海            华东          
0102  南京            华东          
0103  福州            华东          
0104  厦门            华东          
0401  北京            华北          
0402  天津            华北          
0403  大连            华北          
0404  沈阳            华北
# hint

该语句用于提出给定的方案到优化器(Optimizer),使其按照此方案生成语句的执行计划,详情请查阅hint

# distinct

该语句用于对查询结果进行过滤设置,对查询结果中重复的多条记录只返回一条记录。

# select_list

该语句用于指定查询语句中要查询的列项,例如FROM后面某个表的某个列字段(Column)等。查询多项以,分隔。*表示查询FROM后面所有表|视图|AC|子查询等的所有列。

AS alias用于定义别名,AS可省略。

# expr_clause

指定具体的列项,可以为:

  • query_block:将一个查询块语句的结果作为列项
  • expr:表达式
  • CASE语句
  • udt_expr:对于Object UDT列,udt_expr格式为表别名.属性名;对于Varray UDT和Nested Table UDT列,udt_expr格式为表别名.*。详细使用方法和示例见用户自定义类型中描述。

示例(单机部署)

SELECT SYSDATE 查询日期,
a.branch_name 机构,
b.area_name 区域,
(SELECT DHQ FROM area WHERE area_no='10') 默认总部
FROM branches a, area b
WHERE a.area_no IN (SELECT area_no FROM area )
AND a.area_no=b.area_no;
查询日期                     机构          区域                默认总部      
-------------------------------- --------------- --------------------- ------
2022-01-10 10:24:35              上海          华东                          
2022-01-10 10:24:35              南京          华东                          
2022-01-10 10:24:35              福州          华东                          
2022-01-10 10:24:35              厦门          华东                          
2022-01-10 10:24:35              北京          华北                          
2022-01-10 10:24:35              天津          华北                          
2022-01-10 10:24:35              大连          华北                          
2022-01-10 10:24:35              沈阳          华北                          
2022-01-10 10:24:35              成都          华西                          
2022-01-10 10:24:35              长沙          华中

示例(分布式部署)

SELECT SYSDATE 查询日期,
a.branch_name 机构,
b.area_name 区域,
'ShenZhen' 默认总部
FROM branches a, area b
WHERE a.area_no IN ('01','02','03')
AND a.area_no=b.area_no;
查询日期        机构              区域                 默认总部         
------------------- ------------------- ---------------------- ----------
2022-01-23          上海              华东                 ShenZhen     
2022-01-23          南京              华东                 ShenZhen     
2022-01-23          福州              华东                 ShenZhen     
2022-01-23          厦门              华东                 ShenZhen     
2022-01-23          成都              华西                 ShenZhen

# case_clause

case_clause = CASE { simple_case_expression | searched_case_expression } [ else_clause ] END.

CASE表达式相当于编程语言中的IF ELSE,其写法有两种:Simple Case和Searched Case。

simple_case_expression = expr { WHEN comparison_expr THEN return_expr }.

searched_case_expression = { WHEN condition THEN return_expr }.

else_clause = ELSE else_expr.

在Simple Case写法中,系统搜寻第一个与expr等值的comparision_expr,并返回相对应的return_expr。如果没有等值的comparision_expr,并且存在ELSE子句,则返回else_expr。否则返回NULL。

在Searched Case写法中,系统从左至右依次检查condition,并返回第一个相对应的return expr。如果所有condition皆为FALSE,并且存在ELSE子句,则返回else_expr。否则返回NULL。

CASE语句中出现的所有表达式(else_expr/return_expr/expr/comparison_expr,前缀名仅用于区分)均为通用表达式,其值在同一个语句中必须同属下列几种大类之一:

  • 数值型,优先度从低到高的排序为:TINYINT、SMALLINT、INT、BIGINT、NUMBER、FLOAT、DOUBLE。
  • 字符型,优先度从低到高的排序为:CHAR、VARCHAR。
  • 日期时间型,优先度从低到高的排序为:DATE、TIMESTAMP。
  • 其他数据类型归成一个大类。

在每个大类中,表达式的值将先统一为优先度高的数据类型,再进行匹配或返回。因此CASE表达式返回值的数据类型为所有表达式中优先度最高的类型。

示例

SELECT employee_name 姓名, CASE sex
WHEN '0' THEN '女'
WHEN '1' THEN '男'
END 性别
FROM employees;
姓名        性别
------------- ------
Mask          男  
John          男  
Anna          女  
Jack          男  
Jim           男
 
--存在未列出的条件结果时,返回NULL                                                          
SELECT employee_name 姓名, CASE sex
WHEN '2' THEN '女'
WHEN '1' THEN '男'
END 性别
FROM employees;
姓名        性别
------------- ------
Mask          男  
John          男  
Anna               
Jack          男  
Jim           男  
 
--对所有不在列出条件的情况,给定默认值
SELECT employee_name 姓名, CASE sex
WHEN '2' THEN '女'
WHEN '1' THEN '男'
ELSE '未知'
END 性别
FROM employees;
姓名        性别   
------------- ---------
Mask          男     
John          男     
Anna          未知  
Jack          男     
Jim           男

# table_reference

该语句用于指定FROM语句中对查询对象的定义。

YashanDB支持为查询对象分别定义别名,查询对象可为如下类型:

  • query_name:表、视图、同义词、分区、子分区、AC等
  • subquery:一个子查询的结果
  • dblink:远端表

查询对象不能同时存在行存表和列存表,否则返回错误。

# query_name

该语句用于指定表、视图、同义词、分区、子分区、AC等查询对象。

partition_extension_clause

对表分区和子分区的指定方式包括:

  • 根据名称:直接获取分区或子分区对象。
  • 根据提供的键值(Key Value): 将键值与表分区界值比较,计算得到分区或子分区对象,键值可为界值范围内的任意值。键值的个数须与分区列个数对应,指定子分区时键值个数须与分区列和子分区列个数对应,多项用,分隔。

Note

分布式样例表中未指定分区名,由数据库定义分区名及子分区名,可通过查看DBA_TAB_PARTITIONS/DBA_TAB_SUBPARTITIONS查看分区及子分区信息。

示例

-- 执行如下语句查询分区名称
SELECT partition_name
FROM DBA_TAB_PARTITIONS
WHERE table_name='SALES_INFO';
PARTITION_NAME
----------------------------------------------------------------
P_SALES_INFO_1
P_SALES_INFO_2
P_SALES_INFO_3

-- sales_info为一张分区表,可指定分区来查询其数据
SELECT year,month,branch,product,quantity,amount,salsperson FROM sales_info PARTITION (P_SALES_INFO_1);
YEAR  MONTH BRANCH PRODUCT      QUANTITY      AMOUNT SALSPERSON
----- ----- ------ --------- ----------- ----------- -------------
2021  10    0402   11001              20         300

SELECT year,month,branch,product,quantity,amount,salsperson FROM sales_info PARTITION FOR('0102');
YEAR  MONTH BRANCH PRODUCT      QUANTITY      AMOUNT SALSPERSON
----- ----- ------ --------- ----------- ----------- -------------
2001  01    0201   11001              30         500 0201010011
2000  12    0102   11001              20         300
2015  03    0102   11001              20         300

-- 获得sales_info表的子分区名称
SELECT partition_name,subpartition_name 
FROM DBA_TAB_SUBPARTITIONS
WHERE table_name='SALES_INFO';
-- 以下输出以单机为例
PARTITION_NAME               SUBPARTITION_NAME
---------------------------- ---------------------------------------
P_SALES_INFO_1               P_SALES_INFO_1_SP_SALES_INFO_1
P_SALES_INFO_1               P_SALES_INFO_1_SP_SALES_INFO_2
P_SALES_INFO_1               P_SALES_INFO_1_SP_SALES_INFO_3
P_SALES_INFO_2               P_SALES_INFO_2_SP_SALES_INFO_1
P_SALES_INFO_2               P_SALES_INFO_2_SP_SALES_INFO_2
P_SALES_INFO_2               P_SALES_INFO_2_SP_SALES_INFO_3
P_SALES_INFO_3               P_SALES_INFO_3_SP_SALES_INFO_1
P_SALES_INFO_3               P_SALES_INFO_3_SP_SALES_INFO_2
P_SALES_INFO_3               P_SALES_INFO_3_SP_SALES_INFO_3

-- 选择其中一个子分区指定查询数据
SELECT year,month,branch,product,quantity,amount,salsperson FROM sales_info SUBPARTITION (P_SALES_INFO_1_SP_SALES_INFO_2);
YEAR  MONTH BRANCH PRODUCT      QUANTITY      AMOUNT SALSPERSON
----- ----- ------ --------- ----------- ----------- -------------
2021  10    0402   11001              20         300

指定切片查询

对于LSC表,可以通过指定Slice ID查询某个切片的数据,具体规则为:

  • 只能对LSC表指定切片查询,否则报错。
  • 对于分区表,指定切片的同时,必须指定分区,否则报错。
  • Slice ID为0表示查询可变数据区(MCOL)的数据。
  • 通过查询V$LSC_SLICE_STAT视图可获得某张LSC表(分区)包含的所有Slice ID。

示例(LSC表)

-- orders_info表的分区中存在如下稳态数据
SELECT order_no,product_no,area,branch,order_date,salesperson,id FROM orders_info PARTITION(p_orders_info_1);
ORDER_NO          PRODUCT_NO AREA  BRANCH ORDER_DATE            SALESPERSON            ID
----------------- ---------- ----- ------ --------------------- ------------- -----------
20010102020001    11001      01    0101   2023-04-05            0201010011            300
20210102020002    11001      04    0401   2023-04-05            0201008003            400
20210102020002    11002      05    0501   2023-04-05            0201010011            200
20210102020002    10001      01    0102   2023-04-05            0201008003            100

-- 新插入orders_info表一条数据,将存储在可变数据区
INSERT INTO orders_info VALUES ('20210102020003','10002','03','0701',SYSDATE-400,'0201008003',900);

-- 查询V$LSC_SLICE_STAT视图获得Slice ID
SELECT utp.table_name, utp.partition_name, lss.slice_id
FROM USER_TAB_PARTITIONS utp, V$LSC_SLICE_STAT lss, USER_OBJECTS uo
WHERE utp.table_name = 'ORDERS_INFO'
AND uo.subobject_name = utp.partition_name
AND uo.object_id = lss.obj;
TABLE_NAME        PARTITION_NAME        SLICE_ID 
----------------- --------------------  ---------- 
ORDERS_INFO       P_ORDERS_INFO_1       0
ORDERS_INFO       SYS_P12               0
ORDERS_INFO       SYS_P13               0

-- 查询MCOL数据
SELECT order_no,product_no,area,branch,order_date,salesperson,id FROM orders_info PARTITION(p_orders_info_1) SLICE (0);
ORDER_NO          PRODUCT_NO AREA  BRANCH ORDER_DATE            SALESPERSON            ID
----------------- ---------- ----- ------ --------------------- ------------- -----------
20010102020001    11001      01    0101   2023-04-05            0201010011            300
20210102020002    11001      04    0401   2023-04-05            0201008003            400
20210102020002    11002      05    0501   2023-04-05            0201010011            200
20210102020002    10001      01    0102   2023-04-05            0201008003            100
20210102020003    10002      03    0701   2023-04-05            0201008003            500

table_collection_expression

该语句可用于执行如下两种情况:

  • UDT
  • 表函数

UDT

可用于查询Varray UDT或Nested Table UDT数据,UDT数据类型的详细描述请参考用户自定义类型

执行此类查询时,collection_expression为一个Varray对象或Nested Table对象,语法格式为表别名.UDT列名。

Varray对象依赖于所在表,因此应保证执行计划最先访问主表。

示例(HEAP表)

DROP TABLE IF EXISTS city_intro;
CREATE OR REPLACE TYPE brc_array AS VARRAY(20) OF CHAR(15);
/
CREATE OR REPLACE TYPE city_table AS TABLE OF CHAR(10);
/

--创建包含Varray UDT和Nested Table UDT列的city_intro表
CREATE TABLE city_intro (id INT, branches brc_array, citys city_table)
NESTED TABLE citys STORE AS nt_citys;

--插入数据
INSERT INTO city_intro 
VALUES (1,
        brc_array('branch01','branch02','branch03'),
        city_table('shenzhen','guangzhou','dongguan'));
                  
--查询数据
SELECT /*+ LEADING(c) */ c.id, cbranch.*, ccity.*
FROM city_intro c,
TABLE(c.branches) cbranch,
TABLE(c.citys) ccity;
          ID COLUMN_VALUE      COLUMN_VALUE  
------------ ----------------- ------------- 
           1 branch01          shenzhen     
           1 branch02          shenzhen     
           1 branch03          shenzhen     
           1 branch01          guangzhou    
           1 branch02          guangzhou    
           1 branch03          guangzhou    
           1 branch01          dongguan     
           1 branch02          dongguan     
           1 branch03          dongguan  

表函数

可使用表函数进行查询,表函数的具体描述可见内置函数章节中内置表函数部分描述。

执行此类查询时,collection_expression为所需表函数。

# subquery

该语句用于指定一个子查询结果。

示例(单机、共享集群部署)

-- 为sales_info表创建一个同义词,也可指定分区来查询其数据
CREATE SYNONYM sy_sales_info FOR sales_info;
SELECT year,month,branch,product,quantity,amount,salsperson FROM sy_sales_info PARTITION(p_sales_info_1);
YEAR  MONTH BRANCH PRODUCT      QUANTITY      AMOUNT SALSPERSON
----- ----- ------ --------- ----------- ----------- -------------
2021  10    0402   11001              20         300
 
-- 用于子查询中
SELECT b.branch_name 机构,
SUM(s.amount) 销售额
FROM branches b,
(SELECT * FROM sy_sales_info PARTITION(p_sales_info_2)) s
WHERE b.branch_no=s.branch
GROUP BY b.branch_name;
机构                                                             销售额
---------------------------------------------------------------- -----------
成都                                                                   500
南京                                                                   600

# sample_clause

SAMPLE用于从表中按比例抽取一个随机样本,后续WHERE条件将基于这个样本数据而不是整张表数据。

本语句可作用于单个物理表,或者基于单个物理表的视图,对于多表连接的视图执行SAMPLE将会报错。

分布式部署中不可使用本语句。

sample_percent
指定样本所占的百分比,其值必须为一个[0.000001, 100]区间的常量数值(支持可以隐式转换为数值的其他类型)。此百分比代表采样时每个数据行被选为样本的概率,该概率为统计学上的概念,这意味着系统并不会精确返回基于sample_percent计算出的行数的记录。

SEED seed_value
本语句定义一个种子值,系统将尝试对相同的种子值返回相同的样本,未指定本语句时系统每次返回的样本是随机的。seed_value的值必须为一个[0, 4294967295]区间的常量整数(支持可以隐式转换为数值的其他类型,支持将小数截断为整数),其中大于4294967295的数值按4294967295处理。

示例(HEAP表)

-- 随机返回样本
SELECT area_no,area_name,DHQ FROM area SAMPLE(60);
AREA_NO AREA_NAME                 DHQ                   
------- ------------------------- --------------------- 
03      华南                    Guangzhou            
04      华北                    Beijing              
05      华中                    Wuhan  

-- 指定相同seed返回相同样本
SELECT area_no,area_name,DHQ FROM area SAMPLE(60) SEED(7);
AREA_NO AREA_NAME               DHQ                   
------- ----------------------- --------------------- 
01      华东                  Shanghai             
02      华西                  Chengdu              
03      华南                  Guangzhou            
04      华北                  Beijing              
05      华中                  Wuhan

SELECT area_no,area_name,DHQ FROM area SAMPLE(60) SEED(10);
AREA_NO AREA_NAME              DHQ                   
------- ---------------------- --------------------- 
01      华东                 Shanghai             
03      华南                 Guangzhou            
04      华北                 Beijing   

SELECT area_no,area_name,DHQ FROM area SAMPLE(60) SEED(7);
AREA_NO AREA_NAME               DHQ                   
------- ----------------------- --------------------- 
01      华东                  Shanghai             
02      华西                  Chengdu              
03      华南                  Guangzhou            
04      华北                  Beijing              
05      华中                  Wuhan

# flashback_query_clause

YashanDB支持对近期历史数据的追溯查询,例如某条记录在某个时间点已被UPDATE或DELETE更改,但通过flashback_query_clause可以获得该时间点之前一段时间范围内(这取决于undo空间的清理规则),该条记录被更改之前的数据值。

可以使用SCN号或TIMESTAMP标签来定位到想要追溯查询的行记录。

系统在记录TIMESTAMP时使用的是北京时间。

分布式部署中不可使用本语句。

示例(单机、共享集群部署)

-- area表中存在的一条记录
SELECT area_no,area_name,DHQ FROM area WHERE area_no='03';
AREA_NO AREA_NAME          DHQ         
------- ------------------ ------------
03      华南             Guangzhou       
 
-- 获取当前时间
SELECT SYSTIMESTAMP res FROM dual;
RES                                            
----------------------------------------------------------------
2023-12-17 14:14:08.498126     
 
-- 删除此条记录并提交
DELETE FROM area WHERE area_no='03';
COMMIT;
SELECT area_no,area_name,DHQ FROM area WHERE area_no='03';
AREA_NO AREA_NAME          DHQ         
------- ------------------ ------------
 
 
-- 利用flashback可查询到快照历史数据
SELECT area_no,area_name,DHQ FROM area AS OF TIMESTAMP TIMESTAMP('2023-12-17 14:14:08.498126')
WHERE area_no='03';
AREA_NO AREA_NAME          DHQ         
------- ------------------ ------------
03      华南             Guangzhou     

# pivot_clause

PIVOT用于行列转换,通过从结果集中抽出FOR后的列,根据IN里的常量表达式列表构成新的投影列,并结合聚合函数输出对应聚合列的结果。

PIVOT可以对任意结果集(即单表查询或多表连接的结果)进行操作。

pivot_clause前查询的结果集若为多表,只允许使用JOIN ON进行显式join,不允许使用逗号进行隐式join。

IN后的列名的长度大于64字符时,输出的列名会自动截取长度64后的字符。

column_expression

必须为简单列名,即不允许使用table.column的形式。

const_expression

必须为常量表达式。

aggregate_expression

aggregate_expression的聚合列与FOR后的列必须出现在查询的投影列中。

aggregate_expression中使用的聚合函数不允许出现GROUPING_ID、GROUP_ID、GROUPING函数、PERCENTILE_CONT函数、MEDIAN函数以及GROUP_CONCAT的相关函数(GROUP_CONCAT、WM_CONCAT、LISTAGG、STRING_AGG),非聚合函数或无聚合函数时报错。

aggregate_expression的聚合列与IN后的列的别名长度上限为64字符,超过64字符时报错。

示例

-- 抽取区域编号列,并统计对应区域的数量
SELECT * FROM (SELECT branch_name, area_no FROM branches) 
PIVOT(
  COUNT(branch_name)
  FOR area_no 
  IN ('01' AS 华东, '02' AS 华西, '03' AS 华南, '04' AS 华北, '05' AS 华中)
);
华东                华西                华南                华北                华中
--------------------- --------------------- --------------------- --------------------- ---------------------
                    4                     1                     0                     4                     1


-- 显示华东和华西地区最大编号与城市数量
SELECT * FROM (SELECT branch_no, branch_name, area_no FROM branches) 
PIVOT(
  COUNT(branch_name) 城市数量,
  MAX(branch_no) 最大编号 
  FOR area_no 
  IN ('01' AS 华东, '02' AS 华西)
);
华东_城市数量 华东_最大编号   华西_城市数量 华西_最大编号
--------------------- ------------------- --------------------- -------------------
                    4 0104                                    1 0201


# join_clause

该语句用于将多个表、视图、AC和子查询进行连接(Join)查询,有如下几种Join方式:

  • 显示指定JOIN关键字进行连接查询,并通过指定INNER|OUTER关键字分为内连接(Inner Join)和外连接(Outer Join)两种方式,未指定时缺省为内连接。
  • 在FROM后面将各表、视图、子查询以,分隔,而在WHERE后指定条件以获取交叉连接后的查询结果,此种方式等同于内连接。
  • 在WHERE后条件子句中指定(+)操作符,或者在JOIN ON后子句中指定(+)操作符,表示外连接。

分布式部署中遵循如下规则:

  • 多表连接查询分布式系统视图和DBA视图、系统表时,使用的是各节点本地数据。
  • 分布式视图仅允许和分布式视图、DBA视图、USER视图、ALL视图、系统表和单机视图进行多表查询。
  • 多表查询分布式系统视图时,如使用order by、group by、窗口函数、聚集函数和limit,都仅在各节点内生效,最终结果可能不满足有序等约束。

# inner_cross_join_clause

内连接查询是一种交叉连接查询方式,对A和B进行内连接查询时,将A的每一行与B的每一行按照连接条件进行比较,返回的是A、B均满足条件的结果。其中:

  • INNER:产生的结果是AB的交集。此时必须在后面指定ON条件。
  • CROSS:产生的结果是AB的笛卡尔积。此时不可以在后面指定ON条件。

示例(分布式部署)

-- INNER JOIN,INNER关键字可省略
SELECT b.branch_name, a.area_name
FROM branches b
INNER JOIN area a
ON a.area_no = b.area_no
WHERE b.branch_no LIKE '01%';
BRANCH_NAME       AREA_NAME                                                     
----------------- --------------------
上海               华东                                                       
南京               华东                                                       
福州               华东                                                       
厦门               华东                                                                  
 
-- 等同于上面的内连接
SELECT b.branch_name, a.area_name
FROM branches b, area a
WHERE a.area_no=b.area_no AND b.branch_no LIKE '01%';
BRANCH_NAME       AREA_NAME                                                     
----------------- --------------------
上海               华东                                                       
南京               华东                                                       
福州               华东                                                       
厦门               华东                                                       
 
-- CROSS JOIN
SELECT b.branch_name, a.area_name
FROM branches b
CROSS JOIN area a
WHERE b.branch_no LIKE '01%';
BRANCH_NAME       AREA_NAME                                                     
----------------- --------------------
上海               华东                                                       
上海               华西                                                       
上海               华南                                                       
上海               华北                                                       
上海               华中                                                       
南京               华东                                                       
南京               华西                                                       
南京               华南                                                       
南京               华北                                                       
南京               华中                                                       
福州               华东                                                       
福州               华西                                                       
福州               华南                                                       
福州               华北                                                       
福州               华中                                                       
厦门               华东                                                       
厦门               华西                                                       
厦门               华南                                                       
厦门               华北                                                       
厦门               华中           

-- 分布式视图查询                                                
SELECT * FROM (SELECT GROUP_ID,GROUP_NODE_ID,NAME,VALUE,DEFAULT_VALUE,IS_DEPRECATED FROM DV$SYSTEM_PARAMETER WHERE GROUP_ID=0);
GROUP_ID GROUP_NODE_ID NAME          VALUE         DEFAULT_VALUE           IS_DEPRECATED
-------- ------------- ------------- ------------- ----------------------- -------------

# outer_join_clause

外连接查询指定了一张主表,对主表中的记录不要求在对方表里能找到匹配记录,而是全部做为有效结果返回,在返回结果集中对于对方未找到匹配值的列项设为NULL值。外连接需要显式地指定左连接或右连接。

outer_join_type

  • LEFT [OUTER] JOIN

    左连接查询,当A LEFT OUTER JOIN B时,以A为主表,A的每一行记录均会出现在查询结果集中,对于在B中匹配不到的值设为NULL。

  • RIGHT [OUTER] JOIN

    右连接查询,当A RIGHT OUTER JOIN B时,以B为主表,B的每一行记录均会出现在查询结果集中,对于在A中匹配不到的值设为NULL。

  • FULL [OUTER] JOIN

    全连接查询,当A FULL OUTER JOIN B时,A和B的每一行记录均会出现在查询结果集中,对于在A中和B中匹配不到的值设为NULL。

示例

--左连接,对于branches表中的武汉,未在area表中匹配到对应的地区记录,则在结果集中将地区设为NULL
SELECT b.branch_name, a.area_name
FROM branches b
LEFT OUTER JOIN area a
ON a.area_no = b.area_no
WHERE b.branch_no LIKE '01%' OR b.branch_no LIKE '05%';
BRANCH_NAME        AREA_NAME     
------------------ ---------------
上海             华东            
南京             华东            
福州             华东            
厦门             华东            
武汉                             
长沙             华中            
                                   
--右连接,对于area表中的华南地区,未在branches表中匹配到下属的城市记录,则在结果集中将城市设为NULL
SELECT b.branch_name, a.area_name
FROM branches b
RIGHT OUTER JOIN area a
ON a.area_no = b.area_no
WHERE a.area_no IN ('01','03','05');
BRANCH_NAME         AREA_NAME    
------------------- -------------
厦门              华东           
福州              华东           
南京              华东           
上海              华东           
                    华南         
长沙              华中
                                   
--全连接,对于area表中的华南地区,未在branches表中匹配到下属的城市记录,则在结果集中将城市设为NULL
SELECT b.branch_name, a.area_name
FROM branches b
FULL OUTER JOIN area a
ON a.area_no = b.area_no
WHERE a.area_no IN ('01','03','05');
BRANCH_NAME         AREA_NAME    
------------------- -------------
厦门              华东           
福州              华东           
南京              华东           
上海              华东
                    华南    
长沙              华中            

(+) operator

YashanDB支持通过(+)操作符指定外连接。

(+)可以使用在where_clause和join_clause的condition子句的比较条件中。

在比较条件中,(+)作用于某一个列字段上,表示如果比较条件无法满足,则对该列对应的表进行补空,与外连接同义,即比较条件对端所涉及的表(可能为一个或多个)对该表进行外连接。

(+)操作符的使用限制如下:

  • (+)操作符只能作用于列字段上,不可以作用于表达式。
  • 在where_clause条件中使用(+)操作符时,不能与ANSI标准的Join语法一起使用。
  • 在join_clause条件中使用(+)操作符时,连接的两个表必须为普通表,不能为多表Join之后的中间结果。
  • 不允许使用(+)操作符对两个表互相外连接。
  • 某个条件中存在(+)操作符时,该条件的同级条件之间只允许使用AND关键词连接。

示例

--(+)作用于area表的area_no上,意为branches表对area表进行左外连接,结果集对area表中没有匹配上的数据进行补空。
--对于branches表中的武汉,未在area表中匹配到对应的地区记录,则在结果集中将地区设为NULL
SELECT a.area_name,b.branch_name 
FROM area a, branches b
WHERE b.area_no = a.area_no(+)
AND (b.branch_no LIKE '01%' OR b.branch_no LIKE '05%');
AREA_NAME        BRANCH_NAME  
---------------- -------------
华东           上海           
华东           南京           
华东           福州           
华东           厦门           
                 武汉         
华中           长沙    

SELECT a.area_name, b.branch_name
FROM branches b
JOIN area a
ON b.area_no = a.area_no(+)
WHERE b.branch_no LIKE '01%' OR b.branch_no LIKE '05%';
AREA_NAME        BRANCH_NAME  
---------------- -------------
华东           上海           
华东           南京           
华东           福州           
华东           厦门           
                 武汉         
华中           长沙    

# where_clause

该语句用于指定查询的condition

# hierarchical_query_clause

该语句用于指定层次(父子)关系的条件查询。通过START WITH获取根数据,通过 CONNECT BY指定父子关系,即可以将所有满足层次关系的数据全部查找出来。

本语句不适用于列存表。

YashanDB提供如下虚拟列(非表结构中定义的列,但可以被当作列字段进行查询)供层次关系查询中使用:

  • LEVEL:LEVEL表示层次查询形成的树结构中的当前层数。该列值一直从1开始,即START WITH对应的数据的层数一直为1,之后子孙节点的LEVEL值依次递增。

  • CONNECT_BY_ISLEAF:CONNECT_BY_ISLEAF表示当前数据是否是层次查询形成的树结构中的叶子节点。若是叶子节点值为1,否则为0。

  • CONNECT_BY_ISCYCLE:CONNECT_BY_ISCYCLE表示在层次查询中当前数据是否会导致形成环,即根据层次关系,当前层数据是否存在其叶子节点也是其父节点。该列只有在同时指定NOCYCLE关键字时才有意义,当前数据会导致形成环则结果为1,否则为0。

YashanDB提供如下标识符用来指定层次关系中的某个节点属性:

  • PRIOR:PRIOR操作符之后的参数将作为层次查询中的父节点。参数不能为虚拟列、层次查询函数、操作符、伪列及子查询。PRIOR通常应用于connect_by_clause中,且不能在start_with_clause中使用。
  • CONNECT_BY_ROOT:CONNECT_BY_ROOT操作符之后的参数将作为在层次查询中的根节点。参数不能为虚拟列、层次查询函数、操作符、伪列及子查询。CONNECT_BY_ROOT不能在connect_by_clause和start_with_clause中使用。

YashanDB提供如下层次查询中的专用函数:

  • SYS_CONNECT_BY_PATHSYS_CONNECT_BY_PATH(col_name, delimiter),其中delimiter 表示分隔符,只能使用字面量。该函数将获取从根节点到当前节点的路径上所有节点名为col_name的值,中间用delimiter进行分隔开。SYS_CONNECT_BY_PATH不能在connect_by_clause、start_with_clause和group_by_clause中使用。

# connect_by_clause

通过CONNECT BY后跟的condition,指定层次关系查询条件,其中必须至少有一个条件用于指定父子关系,该条件通过PRIOR标识符识别。

NOCYCLE

当根据指定的条件关系获得的查询结果存在环时,如不指定NOCYCLE,系统将返回错误,指定时系统将忽略环的问题,仍返回所有数据。

# start_with_clause

通过START WITH后跟的condition,指定以满足该条件的数据作为层次关系中根节点,其LEVEL为1。condition中不允许使用rownum。

本语句可省略,则表示将所有数据均作为根节点进行层次关系查询。

使用本语句时,其前面或后面必须存在connect_by_clause,不能独立使用。

# order_siblings_by_clause

对同一父节点下的同一层次数据,通过ORDER SIBLINGS BY指定排序列和排序规则进行排序,本语句功能同order_by_clause。

使用本语句时,其前面必须存在connect_by_clause,不能独立使用。

示例(HEAP表)

--建立包含层次关系的地区表area_info
CREATE TABLE area_info (id INT, father_id INT, area_name VARCHAR(20));
INSERT INTO area_info VALUES(1,   0, '广东');
INSERT INTO area_info VALUES(755, 1, '深圳');
INSERT INTO area_info VALUES(756, 755, '龙华');
INSERT INTO area_info VALUES(757, 755, '福田');
INSERT INTO area_info VALUES(2,   0, '浙江');
INSERT INTO area_info VALUES(571, 2, '杭州');
COMMIT;

--显示地区层次关系(指定根节点)   
SELECT id, father_id, LEVEL,
CONNECT_BY_ROOT area_name AS name, 
SYS_CONNECT_BY_PATH(area_name, '/') path  
FROM area_info  
CONNECT BY id<>757 
AND PRIOR id = father_id 
START WITH father_id = 0;  
   ID    FATHER_ID      LEVEL NAME             PATH               
----- ------------ ---------- ---------------- ------------------ 
    1            0          1 广东           /广东               
  755            1          2 广东           /广东/深圳          
  756          755          3 广东           /广东/深圳/龙华     
    2            0          1 浙江           /浙江               
  571            2          2 浙江           /浙江/杭州     
  
--显示地区层次关系(不指定根节点,则所有数据都将作为根节点)   
SELECT id, father_id, LEVEL,
CONNECT_BY_ROOT area_name AS name, 
SYS_CONNECT_BY_PATH(area_name, '/') path  
FROM area_info  
CONNECT BY id<>757 
AND PRIOR id = father_id;  
    ID    FATHER_ID      LEVEL NAME          PATH             
------ ------------ ---------- ------------- ---------------- 
     1            0          1 广东        /广东             
   755            1          2 广东        /广东/深圳        
   756          755          3 广东        /广东/深圳/龙华   
   755            1          1 深圳        /深圳             
   756          755          2 深圳        /深圳/龙华        
     2            0          1 浙江        /浙江             
   571            2          2 浙江        /浙江/杭州        
   571            2          1 杭州        /杭州             
   756          755          1 龙华        /龙华        
   757          755          1 福田        /福田

--显示节点是否为叶子节点及是否出现循环  
SELECT id, father_id, LEVEL, 
CONNECT_BY_ISLEAF AS leaf, CONNECT_BY_ISCYCLE AS iscycle  
FROM area_info  
CONNECT BY NOCYCLE id<>757 
AND PRIOR id = father_id 
START WITH father_id = 0;  
    ID    FATHER_ID     LEVEL LEAF      ISCYCLE     
------ ------------ --------- --------- ----------- 
     1            0         1 false     false      
   755            1         2 false     false      
   756          755         3 true      false      
     2            0         1 false     false      
   571            2         2 true      false      

--按指定列对同层次数据排序  
SELECT  id, father_id, LEVEL,
CONNECT_BY_ROOT area_name AS name, 
SYS_CONNECT_BY_PATH(area_name, '/') path    
FROM area_info  
CONNECT BY PRIOR id = father_id START WITH father_id = 0  
ORDER SIBLINGS BY id DESC;  
    ID    FATHER_ID    LEVEL NAME         PATH            
------ ------------ -------- ------------ --------------- 
     2            0        1 浙江       /浙江            
   571            2        2 浙江       /浙江/杭州       
     1            0        1 广东       /广东            
   755            1        2 广东       /广东/深圳       
   757          755        3 广东       /广东/深圳/福田  
   756          755        3 广东       /广东/深圳/龙华  

# group_by_clause

该语句用于对查询结果集进行按条件的分类聚集(Aggregation)。

# group by

在GROUP BY后定义分组列,多个列用,分隔,同时需满足如下规则:

  • 在select_list中出现的查询列或列数据,必须为分组列或列数据的子集。

列是子集:"SELECT col ,COUNT(*) FROM table GROUP BY col, col2;"

列数据是子集:"SELECT LPAD(col), COUNT(*) FROM table GROUP BY col;"

  • 查询列与分组列里出现函数时,函数的参数必须一致。

此语句错误:"SELECT SUBSTR(col, 1,1), COUNT(*) FROM table GROUP BY SUBSTR(col, 1,2);"

  • 如果同时出现了DISTINCT或ORDER BY子语句,则在它们中出现的列遵循上两条规则。
  • 分组列不能为*(星号),SEQUENCE,子查询及聚集函数。
  • 当分组列为数字时,与ORDER BY不同的是,本语句不会将数字释义成列的位置,而是作为字面量处理。
# having

HAVING子句约束SELECT查询语句中GROUP BY的结果,该约束应用于查询结果中的每个分组,类似WHERE条件应用于select_list。使用规则如下:

  • HAVING子句可以放在GROUP BY子句的前面或后面。
  • HAVING后的condition是一个布尔表达式,语法同WHERE子句中的filter_clause,但是它只能包含分组列、聚集函数(可以与select_list中的聚集函数不一致)、字面量和子查询(子查询中的列不需要为分组列)。其中分布式部署中不可以使用子查询。
  • 如果没有GROUP BY,直接使用HAVING子句,表示该约束作用于整个查询结果,此时select_list和condition中不能出现分组列。

示例(单机部署)

-- 按年份对销售额进行汇总,HAVING后可以指定其他聚集函数,且可以指定对其他表的子查询
SELECT 1,year,month,CONCAT(year,month),SUM(amount) FROM sales_info HAVING MAX(amount)>100 GROUP BY year,month;
           1 YEAR  MONTH CONCAT(YEAR,MONTH)           SUM(AMOUNT)
------------ ----- ----- ------------------ ---------------------
           1 2000  12    200012                               300
           1 2001  01    200101                               500
           1 2015  03    201503                               300
           1 2015  11    201511                               300
           1 2021  05    202105                               600
           1 2021  10    202110                               300
 
SELECT 1,year,month,CONCAT(year,month),SUM(amount)
FROM sales_info
GROUP BY year,month
HAVING SUM(amount)>(SELECT MIN(price*50) FROM product);
           1 YEAR  MONTH CONCAT(YEAR,MONTH)           SUM(AMOUNT)
------------ ----- ----- ------------------ ---------------------
           1 2021  05    202105                               600
 
-- HAVING定义分组列的条件
SELECT 1,year,month,CONCAT(year,month),SUM(amount)
FROM sales_info
GROUP BY year,month
HAVING year>'2015';
           1 YEAR  MONTH CONCAT(YEAR,MONTH)           SUM(AMOUNT)
------------ ----- ----- ------------------ ---------------------
           1 2021  05    202105                               600
           1 2021  10    202110                               300
 
-- 单独使用HAVING语句,定义对整个查询结果的聚集约束条件,此时不能出现分组列
SELECT 1,SUM(amount)
FROM sales_info
HAVING MAX(amount)>100;
           1           SUM(AMOUNT)
------------ ---------------------
           1                  2300

示例(单机、分布式部署)

-- 按年份对销售额进行汇总,HAVING后指定相同聚集函数
SELECT 1,year,month,CONCAT(year,month),SUM(amount)
FROM sales_info
HAVING SUM(amount)>300
GROUP BY year,month;
           1 YEAR  MONTH CONCAT(YEAR,MONTH) SUM(AMOUNT)
------------ ----- ----- ------------------ -----------
           1 2001  01    200101                     500
           1 2021  05    202105                     600
 
SELECT 1,year,month,CONCAT(year,month),SUM(amount)
FROM sales_info
GROUP BY year,month
HAVING SUM(amount)>500;
           1 YEAR  MONTH CONCAT(YEAR,MONTH)           SUM(AMOUNT)
------------ ----- ----- ------------------ ---------------------
           1 2021  05    202105                               600
 
-- HAVING定义分组列的条件
SELECT 1,year,month,CONCAT(year,month),SUM(amount)
FROM sales_info
GROUP BY year,month
HAVING year>'2015';
           1 YEAR  MONTH CONCAT(YEAR,MONTH)           SUM(AMOUNT)
------------ ----- ----- ------------------ ---------------------
           1 2021  05    202105                               600
           1 2021  10    202110                               300
 
-- 单独使用HAVING语句,定义对整个查询结果的聚集约束条件,此时不能出现分组列
SELECT 1,SUM(amount)
FROM sales_info
HAVING SUM(amount)>300;
           1           SUM(AMOUNT)
------------ ---------------------
           1                  2300

# rollup_cube_clause

ROLLUP

该关键字用于指定ROLLUP执行算子,该算子用于拓展GROUP BY聚集功能,区别在于GROUP BY仅返回每个分组的结果,指定了ROLLUP关键字会返回总计和每个分组的结果。

其等价于对ROLLUP后指定的列字段的每个层次级别创建grouping set,如ROLLUP(A,B,C) == GROUPING SETS((A, B, C), (A, B), (A), ())

本语句不适用于HEAP表。

示例(LSC表、TAC表)

SELECT year,product,salsperson,SUM(quantity) FROM sales_info GROUP BY ROLLUP(year,product,salsperson);

YEAR  PRODUCT   SALSPERSON    SUM(QUANTITY) 
----- --------- ------------- ------------- 
2000  11001                              20
2001  11001     0201010011               30
2015  11001                              40
2021  11001                              60
2000  11001                              20
2001  11001                              30
2015  11001                              40
2021  11001                              60
2000                                     20
2001                                     30
2015                                     40
2021                                     60
                                        150

CUBE

该关键字用于指定CUBE执行算子,该算子用于拓展GROUP BY聚集功能,区别在于GROUP BY仅返回每个分组的结果,指定了CUBE关键字会返回所有组合的结果,其中包括每个分组的结果。

其等价于对CUBE后指定的列字段的所有组合创建grouping set,如CUBE(A,B,C) == GROUPING SETS((A, B, C), (A, B), (A, C), (B, C), (A), (B), (C), ())

本语句不适用于HEAP表。

示例(LSC表、TAC表)

SELECT year,product,salsperson,SUM(quantity) FROM sales_info GROUP BY CUBE(year,product,salsperson);

YEAR  PRODUCT   SALSPERSON    SUM(QUANTITY) 
----- --------- ------------- ------------- 
2000  11001                              20
2001  11001     0201010011               30
2015  11001                              40
2021  11001                              60
2000  11001                              20
2001  11001                              30
2015  11001                              40
2021  11001                              60
      11001                             120
      11001     0201010011               30
      11001                             150
2000                                     20
2001            0201010011               30
2015                                     40
2021                                     60
2000                                     20
2001                                     30
2015                                     40
2021                                     60
                                        150
                                        120
                0201010011               30

# grouping_sets_clause

该语句用于指定GROUP BY的分组规则,并将结果聚合,等价于对指定组合执行GROUP BY分组,然后通过UNION ALL将结果联合起来。

本语句不适用于HEAP表。

示例(LSC表、TAC表)

SELECT year,product,salsperson,SUM(quantity) FROM sales_info GROUP BY GROUPING SETS(year,product,salsperson);

YEAR  PRODUCT   SALSPERSON    SUM(QUANTITY) 
----- --------- ------------- ------------- 
                                        120
                0201010011               30
2000                                     20
2001                                     30
2015                                     40
2021                                     60
      11001                             150

# set_oper

集合操作,用于将两个或两个以上的查询结果集组合成单个结果集,包括如下类型:

  • UNION:合并且过滤掉重复值。

  • UNION ALL:合并且保留重复值。

  • INTERSECT:比较两个查询的结果,选择所有查询结果集中相同行的记录,且过滤掉重复值。

  • INTERSECT ALL:比较两个查询的结果,选择所有查询结果集中相同行的记录,且保留重复值。

  • MINUS/EXCEPT:比较两个查询的结果,返回在第一个查询结果集中,但不是第二个查询结果集中的行的记录,且过滤掉重复值。

  • MINUS ALL/EXCEPT ALL:比较两个查询的结果,返回在第一个查询结果集中,但不是第二个查询结果集中的行的记录,且保留重复值。

集合操作中MINUS与EXCEPT完全等价。

进行集合操作的查询结果集的数据类型必须处于下面同一个分类中,否则函数返回Query column mismatch错误:

  • 数值型:按优先度从高到低为DOUBLE、FLOAT、NUMBER、BIGINT、INT、SMALLINT、TINYINT,但BIT类型只能与同类型在同一个操作列表中。
  • 字符型:按优先度从高到低为VARCHAR、CHAR。
  • 日期时间型:按优先度从高到低为TIMESTAMP、DATE、TIME,但INTERVAL YEAR TO MONTH、INTERVAL DAY TO SECOND只能与同类型在同一个参数列表中。
  • 布尔型:BOOLEAN。
  • 其他类型:RAW。

Note

1.同一分类中数据类型不同时,低优先度类型向高优先度类型转换。

2.结果集为常量NULL(查询列为常量NULL所生成的结果集)时,不受上述分类限制。

3.所有集合运算为统一运算优先级,运算顺序以书写顺序为准,可通过双括号()调整想要达到的运算优先级。

示例

-- UNION
SELECT area_no,area_name,DHQ
FROM (SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','02')
      UNION
      SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','03')
     );
AREA_NO AREA_NAME          DHQ      
------- ------------------ ----------
01      华东             Shanghai   
02      华西             Chengdu    
03      华南             Guangzhou  
  
-- UNION ALL
SELECT area_no,area_name,DHQ
FROM (SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','02')
      UNION ALL
      SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','03')
     );
AREA_NO AREA_NAME          DHQ      
------- ------------------ ----------
01      华东             Shanghai   
03      华南             Guangzhou  
01      华东             Shanghai   
02      华西             Chengdu

-- INTERSECT
SELECT area_no,area_name,DHQ
FROM (SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','02')
      INTERSECT
      SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','03')
     );
AREA_NO AREA_NAME          DHQ      
------- ------------------ ----------
01      华东             Shanghai   
  
-- INTERSECT ALL
SELECT area_no,area_name,DHQ
FROM (SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','02')
      INTERSECT ALL
      SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','03')
     );
AREA_NO AREA_NAME          DHQ      
------- ------------------ ----------
01      华东             Shanghai   

-- MINUS
SELECT area_no,area_name,DHQ
FROM (SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','02')
      MINUS
      SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','03')
     );
AREA_NO AREA_NAME          DHQ      
------- ------------------ ----------
02      华西             Chengdu  
  
-- MINUS ALL
SELECT area_no,area_name,DHQ
FROM (SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','02')
      MINUS ALL
      SELECT area_no,area_name,DHQ FROM area WHERE area_no IN ('01','03')
     );
AREA_NO AREA_NAME          DHQ      
------- ------------------ ----------
02      华西             Chengdu   

# order_by_clause

该语句用于对查询结果集按照排序键进行排序。

排序键可以为:

  • expr:表达式
  • position:select_list中的列项位置,默认按此项排序
  • c_alias:select_list中为列项定义的别名

排序键不允许为如下类型:

  • CLOB
  • BLOB
  • NCLOB
  • BIT
  • ROWID
  • CURSOR
  • JSON
  • UDT
# nulls (first|last)

对于空值,用NULLS FIRST|NULLS LAST语句来指定空值排列在最前或最后。当未指定此语句时,对升序排列缺省为NULLS LAST,对降序排列缺省为NULLS FIRST。

# asc|desc

排序方式可指定为升序(ASC)或降序(DESC),未指定则缺省为升序。

示例

SELECT year,branch,SUM(amount) FROM sales_info GROUP BY year,branch ORDER BY SUM(amount)/SUM(quantity);
YEAR  BRANCH           SUM(AMOUNT)
----- ------ ---------------------
2015  0101                     300
2015  0102                     300
2000  0102                     300
2021  0101                     900
2001  0201                     500
 
SELECT year,branch,SUM(amount) FROM sales_info GROUP BY year,branch ORDER BY 2;
YEAR  BRANCH           SUM(AMOUNT)
----- ------ ---------------------
2015  0101                     300
2021  0101                     900
2015  0102                     300
2000  0102                     300
2001  0201                     500

# row_limiting_clause

该语句用于抓取查询结果集的指定行。

# limit

指定抓取的行数。LIMIT满足如下规则:

  • LIMIT可以为一个表达式,表达式的结果必须为一个数字。
  • 查询HEAP表时,LIMIT可以使用标量子查询和绑定参数。
  • 如果LIMIT是负值,则将其视为0。
  • 如果LIMIT是NULL,则不返回任何行。
  • 如果LIMIT大于查询的行数,则返回所有行。
  • 如果LIMIT是小数,则小数部分被截断。
# offset

指定抓取的起始行偏离数,缺省为第一行的偏离数0。OFFSET满足如下规则:

  • OFFSET可以为一个表达式,表达式的结果必须为一个数字。
  • 查询HEAP表时,OFFSET可以使用标量子查询和绑定参数。
  • 如果OFFSET是负值,则将其视为0。
  • 如果OFFSET是NULL,或者大于查询的行数,则不返回任何行。
  • 如果OFFSET是小数,则小数部分被截断。

执行本语句将对优化器产生以下影响:

  • 如果SQL中同时出现了order_by_clause和row_limiting_clause,优化器会将排序计划改写成ORDER BY STOPKEY计划。
  • 如果SQL中同时出现了group_by_clause、distinct、order_by_clause、以及row_limiting_clause,优化器将基于cost可能生成SORT GROUP BY STOPKEY/SORT DISTINCT STOPKEY计划。

示例

-- 从branches表的第一行开始抓取前四行数据
SELECT branch_no,branch_name,area_no,address FROM branches LIMIT 4;
BRANCH_NO BRANCH_NAME         AREA_NO ADDRESS         
--------- ------------------- ------- -----------------
0001      深圳                                        
0101      上海              01      上海市静安区      
0102      南京              01      City of Nanjing   
0103      福州              01                        
 
-- 从branches表的第四行开始抓取前四行数据 
SELECT branch_no,branch_name,area_no,address FROM branches LIMIT 4 OFFSET 3;
BRANCH_NO BRANCH_NAME         AREA_NO ADDRESS         
--------- ------------------- ------- -----------------
0103      福州              01                        
0104      厦门              01      Xiamen            
0401      北京              04                        
0402      天津              04

# offset_fetch_clause

OFFSET FETCH的功能实现和LIMIT OFFSET完全一致,用于抓取查询结果集的指定行。

# rowcount

指定抓取的行数。ROWCOUNT满足如下规则:

  • ROWCOUNT可以为一个表达式,表达式的结果必须为一个数字。
  • 查询HEAP表时,LIMIT可以使用标量子查询和绑定参数。
  • 如果ROWCOUNT是负值,则将其视为0。
  • 如果ROWCOUNT是NULL,则不返回任何行。
  • 如果ROWCOUNT大于查询的行数,则返回所有行。
  • 如果ROWCOUNT是小数,则小数部分被截断。
# offset

指定抓取的起始行偏离数,缺省为第一行的偏离数0。OFFSET满足如下规则:

  • OFFSET可以为一个表达式,表达式的结果必须为一个数字。
  • 查询HEAP表时,OFFSET可以使用标量子查询和绑定参数。
  • 如果OFFSET是负值,则将其视为0。
  • 如果OFFSET是NULL,或者大于查询的行数,则不返回任何行。
  • 如果OFFSET是小数,则小数部分被截断。

示例

-- 从branches表的第一行开始抓取前四行数据
SELECT branch_no,branch_name,area_no,address FROM branches FETCH FIRST 4 ROW ONLY;
BRANCH_NO BRANCH_NAME         AREA_NO ADDRESS         
--------- ------------------- ------- -----------------
0001      深圳                                        
0101      上海              01      上海市静安区      
0102      南京              01      City of Nanjing   
0103      福州              01                        
 
-- 从branches表的第四行开始抓取前四行数据 
SELECT branch_no,branch_name,area_no,address FROM branches OFFSET 3 ROW FETCH FIRST 4 ROW ONLY;
BRANCH_NO BRANCH_NAME         AREA_NO ADDRESS         
--------- ------------------- ------- -----------------
0103      福州              01                        
0104      厦门              01      Xiamen            
0401      北京              04                        
0402      天津              04

# for_update_clause

该语句用于将查询结果集对应的行记录锁定,锁定后其他用户将只能对这些行进行查询,而不能执行锁定或修改等操作,直至当前事务结束。

只能在顶层的SELECT语句中指定FOR UPDATE,子查询和CTE不能指定FOR UPDATE。

本语句不可与DISTINCT,GROUP BY,FLASHBACK,聚集函数,游标表达式等组合使用。

# of column_name

用OF后的列字段名来指定锁定该列字段所在的表,而不需要将FROM后的表都锁定。本语句一般在多表连接情况下使用,指定对多表中特定的某些表执行FOR UPDATE。省略本语句时锁定当前查询中所有涉及到的表对象对应的记录。

通过column_name标识出其所在的表对象,必须是一个真实的列字段名称,不能为别名。

可联合表、视图、同义词,或者在FROM语句中定义的别名,来指定列字段名。

# wait|nowait|skip locked

指定当要锁定的记录里至少有部分行正被其他用户锁定时的操作,缺省为一直等待到那些行被解锁后,再执行操作并返回。

  • WAIT ntimes:等待指定的时间,ntimes的单位为秒,超过此时间后报错。
  • NOWAIT:不等待且不执行操作,直接返回。
  • SKIP LOCKED:跳过被锁定的行,继续执行其他的行。不适用于LSC表的稳态数据。

本语句用来定义行锁时的操作,当出现表锁时,直接进入表锁等待。

示例(单机HEAP表、TAC表)

SELECT TRIM(address) address FROM branches FOR UPDATE;
 
SELECT * FROM area a,branches b WHERE a.area_no=b.area_no AND a.area_no='01' FOR UPDATE OF a.DHQ;         --此时仅锁定area表中的对应行
 
SELECT TRIM(address) address FROM branches FOR UPDATE NOWAIT;
 
SELECT TRIM(address) address FROM branches FOR UPDATE WAIT 10;
 
SELECT TRIM(address) address FROM branches FOR UPDATE SKIP LOCKED;