概述
目录
1 SQL语言概述
1.1 SQL 语言的分类
1.1.1 数据查询语言(DQL)
1.1.2 数据操纵语言(DML)
1.1.3 事务控制语言(TCL)
1.1.4 数据定义语言(DDL)
1.1.5 数据控制语言(DCL)
1.2 SQL 语言的编写规则
2 检索数据
3 分页查询
3.1 三层嵌套方法
3.2 rownum分析
1 SQL语言概述
1.1 SQL 语言的分类
SQL 是关系型数据库的基本操作语言,是数据库管理系统与数据库进行交互的接口。它将数据查询、数据操纵、事务控制、数据定义和数据控制功能集于一身,而这些功能又分别对应着各自的 SQL语言,具体如下:
1.1.1 数据查询语言(DQL)
用于检索数据库中的数据,主要是 SELECT 语句,它在操作数据库的过程中使用最为频繁。
1.1.2 数据操纵语言(DML)
用于改变数据库中的数据,主要包括 INSERT、 UPDATE 和 DELETE 3 条语句。其中, INSERT 语句用于将数据插入到数据库中, UPDATE 语句用于更新数据库中已经存在的数据,而 DELETE 语句则用于删除数据库中已经存在的数据。
1.1.3 事务控制语言(TCL)
用于维护数据的一致性,包括 COMMIT、 ROLLBACK 和 SAVEPOINT 3 条语句。其中, COMMIT语句用于提交对数据库的更改, ROLLBACK 语句用于取消对数据库的更改,而 SAVEPOINT 语句则用于设置保存点。
1.1.4 数据定义语言(DDL)
用于建立、修改和删除数据库对象。比如,可以使用 CREATE TABLE 语句创建表,使用 ALTERTABLE 语句修改表结构,使用 DROP TABLE 语句删除表。
1.1.5 数据控制语言(DCL)
用于执行权限授予和权限收回操作,主要包括 GRANT 和 REVOKE 两条命令。其中, GRANT 命令用于给用户或角色授予权限,而 REVOKE 命令则用于收回用户或角色所具有的权限。
1.2 SQL 语言的编写规则
SQL 关键字不区分大小写,既可以使用大写格式,也可以使用小写格式,或者大小写格式混用
但是:字符值区分大小写。当在 SQL 语句中引用字符值时,必须给出正确的大小写数据,否则,不能得到正确的查询结果。
SQL> select * from scott.emp where job='SALESMAN';
SQL> select * from scott.emp where job='salesman';
分别执行这两条 SELECT 语句,会发现结果不相同,因为查询条件是不相同的。
在 SQL*Plus 环境编写 SQL 语句时, 如果 SQL 语句较短, 则可以将语句放在一行上显示; 如果 SQL语句很长,为了便于用户阅读,则可以将语句分行显示(并且 Oracle 会在除第一行之外的每一行前面自动加上行号),当 SQL 语句输入完毕,要以分号作为结束符。
2 检索数据
用户对表或视图最常进行的操作就是检索数据,检索数据可以通过 SELECT 语句来实现,该语句由多个子句组成,通过这些子句可以完成筛选、投影和连接等各种数据操作,最终得到用户想要的查询结果。该语句的基本语法格式如下:
select {[ distinct | all ] columns | *}
[into table_name]
from {tables | views | other select}
[where conditions]
[group by columns]
[having conditions]
[order by columns]
在上面的语法中,共有 7 个子句,它们的功能分别如下:
- select 子句:用于选择数据表、视图中的列。
- into 子句:用于将原表的结构和数据插入新表中。
- from 子句:用于指定数据来源,包括表,视图和其他 select 语句。
- where 子句:用于对检索的数据进行筛选。
- group by 子句:用于对检索结果进行分组显示。
- having 子句:用于从使用 group by 子句分组后的查询结果中筛选数据行。
- order by 子句:用来对结果集进行排序(包括升序和降序)。
在 Oracle 数据库中,有一个标识行中唯一特性的行标识符,该行标识符的名称为 ROWID。行标识符 ROWID 是 Oracle 数据库内部使用的隐藏列,由于该列实际上并不是定义在表中,所以也称为伪列。伪列 ROWID长度为 18位字符,包含了该行数据在 Oracle数据库中的物理地址。用户使用 DESCRIBE命令是无法查到 ROWID 列的,但是可以在 SELECT 语句中检索到该列。
对于rownum来说它是oracle系统顺序分配为从查询返回的行的编号,返回的第一行分配的是1,第二行是2,依此类推,这个伪字段可以用于限制查询返回的总行数,且rownum不能以任何表的名称作为前缀。
如果想找到从第二行记录以后的记录,当使用rownum>2是查不出记录的,原因是由于rownum是一个总是从1开始的伪列,Oracle 认为rownum> n(n>1的自然数)这种条件依旧不成立,所以查不到记录,但是rownum对于rownum<n((n>1的自然数)的条件认为是成立的,所以可以找到记录。
另外:在使用 SELECT 语句时,对于数字数据和日期数据都可以使用算术表达式。在 SELECT 语句中可以使用算术运算符,包括加(+)、减(-)、乘(*)、除(/)和括号。另外,在 SELECT 语句中不仅可以执行单独的数学运算,还可以执行单独的日期运算以及与列名关联的运算
3 分页查询
3.1 三层嵌套方法
SELECT * FROM ( SELECT A.*, ROWNUM RN
FROM (SELECT * FROM TABLE_NAME) A WHERE ROWNUM <= 40 ) WHERE RN >= 21
其中最内层的查询SELECT * FROM TABLE_NAME表示不进行翻页的原始查询语句。ROWNUM = 21控制分页查询的每页的范围。
上面给出的这个分页查询语句,在大多数情况拥有较高的效率。分页的目的就是控制输出结果集大小,将结果尽快的返回。在上面的分页查询语句中,这种考虑主要体现在WHERE ROWNUM
选择第21到40条记录存在两种方法,一种是上面例子中展示的在查询的第二层通过ROWNUM<= 40来控制最大值,在查询的最外层控制最小值。而另一种方式是去掉查询第二层的WHERE ROWNUM<=40语句,在查询的最外层控制分页的最小值和最大值。
这是,查询语句如下:
SELECT * FROM ( SELECT A.*, ROWNUM RN
FROM (SELECT * FROM TABLE_NAME) A ) WHERE RN BETWEEN 21 AND 40
对比这两种写法,绝大多数的情况下,第一个查询的效率比第二个高得多。
这是由于CBO优化模式下,Oracle可以将外层的查询条件推到内层查询中,以提高内层查询的执行效率。对于第一个查询语句,第二层的查询条件WHERE ROWNUM<= 40就可以被Oracle推入到内层查询中,这样Oracle查询的结果一旦超过了ROWNUM限制条件,就终止查询将结果返回了
而第二个查询语句,由于查询条件BETWEEN 21 AND 40是存在于查询的第三层,而Oracle无法将第三层的查询条件推到最内层(即使推到最内层也没有意义,因为最内层查询不知道RN代表什么)。因此,对于第二个查询语句,Oracle最内层返回给中间层的是所有满足条件的数据,而中间层返回给最外层的也是所有数据。数据的过滤在最外层完成,显然这个效率要比第一个查询低得多。
上面分析的查询不仅仅是针对单表的简单查询,对于最内层查询是复杂的多表联合查询或最内层查询包含排序的情况一样有效。
3.2 rownum分析
对于Oracle的rownum问题,很多资料都说不支持>,>=,=,between……and,只能用以上符号(并非说用>,>=,=,between……and 时会提示SQL语法错误,而是经常是查不出一条记录来,还会出现似乎是莫名其妙的结果来,其实您只要理解好了这个rownum伪列的意义就不应该感到惊 奇,同样是伪列,rownum与rowid可有些不一样,下面以例子说明:
假设某个表t1(c1)有20条记录。如果用select rownum,c1 from t1 where rownum < 10,只要是用小于号,查出来的结果很容易地与一般理解在概念上能达成一致,应该不会有任何疑问的。
可如果用select rownum,c1 from t1 where rownum > 10(如果写下这样的查询语句,这时候在您的头脑中应该是想得到表中后面10条记录),你就会发现,显示出来的结果要让你失望了,也许还会怀疑是不谁删 了一些记录,然后查看记录数,仍然是20条啊?那问题是出在哪呢?
先好好理解rownum的意义吧。因为ROWNUM是对结果集加的一 个伪列,即先查到结果集之后再加上去的一个列 (强调:先要有结果集)。简单的说rownum是对符合条件结果的序列号。它总是从1开始排起的。所以你选出的结果不可能没有1,而有其他大于1的值。所以没办法期望得到下面的结果集:
11 aaaaaaaa
12 bbbbbbb
13 ccccccc……
rownum >10 没有记录,因为第一条不满足去掉的话,第二条的ROWNUM又成了1,所以永远没有满足条件的记录。或者可以这样理解:ROWNUM是一个序列,是oracle数据库从数据文件或缓冲区中读取数据的顺序。它取得第一条记录则rownum值为1,第二条为2,依次类推。如 果你用>,>=,=,between……and这些条件,因为从缓冲区或数据文件中得到的第一条记录的rownum为1,则被删除,接着取下条,可是它的rownum还是1,又被删除,依次类推,便没有了数据。
有了以上从不同方面建立起来的对rownum的概念,那我们可以来认识使用 rownum 的几种现像
- select rownum,c1 from t1 where rownum != 10 为何是返回前9条数据呢?它与 select rownum,c1 from tablename where rownum < 10 返回的结果集是一样的呢?
因为是在查 询到结果集后,显示完第 9 条记录后,之后的记录也都是 != 10,或者 >=10,所以只显示前面9条记录。也可以这样理解,rownum为9后的记录的 rownum为10,因条件为 !=10,所以去掉,其后记录补上,rownum又是10,也去掉,如果下去也就只会显示前面9条记录了 - 为什么 rownum >1 时查不到一条记录,而 rownum >0 或 rownum >=1 却总显示所以的记录因为 rownum 是在查询到的结果集后加上去的,它总是从1开始
- 为什么 between 1 and 10 或者 between 0 and 10 能查到结果,而用 between 2 and 10 却得不到结果原因同上一样,因为 rownum 总是从 1 开始
从上可以看出,任何时候想把 rownum = 1 这条记录抛弃是不对的,它在结果集中是不可或缺的,少了rownum=1 就像空中楼阁一般不能存在,所以你的 rownum 条件要包含到 1,但如果就是想要用 rownum > 10这种条件的话就要用嵌套语句,把rownum先生成,然后对他进行查询。
select *
from (selet rownum as rn,t1.* from a where ……)
where rn >10
一般代码中对结果集进行分页就是这么干的。
另外:rowid与rownum虽都被称为伪列,但它们的存在方式是不一样的,rowid可以说是物理存在的,表示记录在表空间中的唯一位置ID,在 DB中唯一。只要记录没被搬动过,rowid是不变的。rowid相对于表来说又像表中的一般列,所以rowid为条件就不会有 rownum那些情况发生。
rownum和排序:
select rownum ,id,name from student order by name;
ROWNUM ID NAME
---------- ------ ---------------------------------------------------
3 200003 李三
2 200002 王二
1 200001 张一
4 200004 赵四
可以看出,rownum并不是按照name列来生成的序号。系统是按照记录插入时的顺序给记录排的号,rowid也是顺序分配的。为了解决这个问题,必须使用子查询
SQL> select rownum ,id,name from (select * from student order by name);
ROWNUM ID NAME
---------- ------ ---------------------------------------------------
1 200003 李三
2 200002 王二
3 200001 张一
4 200004 赵四
这样就成了按name排序,并且用rownum标出正确序号(由小到大)
最后
以上就是大意电脑为你收集整理的Oracle中SQL语言和分页rownum分析1 SQL语言概述2 检索数据3 分页查询的全部内容,希望文章能够帮你解决Oracle中SQL语言和分页rownum分析1 SQL语言概述2 检索数据3 分页查询所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
发表评论 取消回复