常见的sql分类五种分类
- DDL (Data Definition Language)数据库定义语言
- DML (Data Manipulation Language)数据操作语言
- DCL (Data Control Language)数据控制语言
- TCL (Transactional Control Language) 事务控制语法
- DQL (Data Query Language) 数据查询语言
有时候会把DQL归入到DML中, 因为DML常见的操作是针对数据表中数据的 增删改 ,而查数据的DQL,同样是基于表中数据级别的操作,所以真的把DQL归入到DML也说的过去,但是DQL 单条select语句的复杂程度就远远高于 DDL DML DCL TCL中的任意一种,所以我单独整理出来
Select知识图谱
如何写查询sql?
– 一条查询语句,最容易确定就是 要查询的结果是什么! 也就是 select 后面紧跟的信息;
– 数据从哪里获取? from XXX表
1 | -- 最简单的查询 ,`查询某某某,且表中的字段` 简单的没话说 |
exits 关键字
– exists (查询语言) , 在大数据级别下,它的效率很高!!!
– 查询到结果 true 否则 false
例:查询有员工的部门信息:
1 | select * from DEPT d where exists (select * from emp e where d.DEPTNO=e.DEPTNO); |
虚表/伪标
dual是Oracle中的虚表,它的存在主要是为了补全语法结构(select XXX 后面没有from 的话,报错 ), 里面只有一行一列 存着个X
1 | select 1+1; // 在Oracle中会报错, 而Mysql不会报错 |
伪列 rownum
就是Oracle 特有的用来表示行号的, 默认起始值为1 ,有了结果再 +1
– 练习; 找到员工表中,工资最高的前三名 – 添加上一个条件 rownum<4 就行
1 | -- 空格+新表名 里面的表可以用里面的表的别名 反之不可以 |
– 练习: 查询员工表中的 薪水大于本部门的平均薪水的员工1
2
select * from (select DEPTNO ,avg(sal) avgsal from EMP group by deptno) table1 , emp where emp.DEPTNO=table1.DEPTNO and emp.sal>table1.avgsal;
– 在Oracle中的分页
– mysql中的分页使用limit
– oracle只能使用rownum1
select * from (select rownum hanghao,emp.* from EMP) tt where tt.hanghao between 1 and 5;
伪列 rowid
伪列,每行记录存放的真实物理地址
简单使用,查看emp表里面每行的物理地址
1 | select rowid ,e.* from EMP e; |
– 去除(删除) 表中的重复记录1
2
3
4
5
6
7
8
9create table p(name varchar2(20));
insert into p(name) values ('张三');
insert into p(name) values ('张三');
insert into p(name) values ('李四');
select ROWID from p;
-- 删除,rowid 是从小到大排序的, 我们只要 rowid 小的
delete from p where p.ROWID>(select min(rowid) from p p2 where p.name=p2.name);
并集运算
– 交并补
– 例: 工资大于1500 ,或者20号部门下的员工1
select * from EMP where emp.sal>1500 or EMPNO=20;
– 并集 union 去重并且排序
– 并集 union all 不去重
—差集 minus A minux B A-B
– 注意点:
– 1.列的类型要一致
– 2. 按顺序书写
– 3. 列的数量要一致,如果不一致,会用空值填充
用法
1 | select XXX union , union, minus select XXX; |
别名查询
as关键字可以省略
1 | select ename as "姓 名", sal as 工资 from emp; |
取出重复数据
distinct关键字,在select之后
查询中的四则运算
- select 显示字段 … ; 显示字段如果是数字,Oracle支持四则运算
- 注意:null值, 不可以做四则运算– 解决 nvl(P1,P2)函数,若P1为null,那么取P2的值
1 | select sal*12 + comm from EMP; |
字符串的拼接符
- Oracle中使用 || 进行字符串的拼接
1 | select ENAME from EMP; |
在mysql和oracle 中都有concat(str1,str2)函数,用于字符串拼接
条件查询(where 后面的条件)
- 关系运算符: > >= < <= != <>
- 逻辑运算符: and or not
- 其他运算符:
- like % 匹配多个字符 _ 匹配单个字符
- in
- between..and…
- is null
- is not null
Oracle处理转义字符:escape 'X'
1 | // 查询名字中包含% 的学生的信息 |
排序:order by
- 升序:asc
- 降序:desc
注意点: 排序时,空值默认会排到前面去
解决:1
2
3
4// 默认
select * from emp order by comm desc nulls first;
// 解决:
select * from emp order by comm desc nulls last;
函数
- 单行函数
- 对行中的某一个值进行处理
- 数值函数
- 字符函数
- 日期函数
- 转换函数
- 通用函数
- 对行中的某一个值进行处理
数值函数:
1 | 向上取整 |
字符函数:
- 截取字符串 : substr(“被操作的字符串”,起始索引,长度)
注意点: 不管起始索引从0开始还是1开始,都是从第一个字符开始算
- 获取字符串的长度: length(‘sdfghj’)
注意从1开始计数
- 去除字符串左右两边的空格
trim(‘ ghj ‘)
日期函数
1 | -- 查询今天的日期 |
转换函数
- 转数字: 字符串+数字 = 数字 <==> to_number(‘字符串’)
- 转字符串: to_char(数字/日期,’转换后的格式(yyyy-MM-dd / $999,999,.999)’)
- 字符转日期: to_date(‘字符串’,’准换后的格式’)
通用函数
- nv1(p1,p2) // 如果 p1==null 返回 p2
- nv12(p1,p2,p3) // 如果 p1==null 返回p3 否则返回 p2
- nullif(p1,p2) // 如果 p1==p2 返回null 否则返回p1
- coalesce(p1,p2….) // 返回第一个不为空的值
- 多行函数
- 对所有列进行处理
- max()
- min()
- count()
- sum()
- avg()
- 对所有列进行处理
注意点: avg() 平均值 受空值影响
1 | // 求工资平均值 |
条件表达式
- Oracle 和 Mysql 都有 — case when XXX then XXX
1 | 给表中人取别名,并显示 |
- Oracle 特有的 — decode()
1
2
3
4select decode(ename,'ALLEN','哈哈','名字') from EMP;
-- 如果是 W 改为哈哈 R 改为 哈哈
select decode(ename,'W','哈哈','R','哈哈') from emp;
分组表达式 group by
– 写法1
select 分组条件 分组之后的操作 from 表名 group by 分组的条件 having 条件过滤
– 分组统计所有部门的平均工资,找出平均工资大于2000的部门1
select DEPTNO, avg(SAL) from EMP group by DEPTNO having avg(SAL)>2000;
– 错误写法1
select DEPTNO, avg(SAL) avg from EMP group by DEPTNO having avg>2000;
- 注意点1 : select后面的 是不能 声明 别名的!!!
原因: sql的书写顺序和执行顺序是不同的
– 书写顺序1
2
3
4
5select [分组条件] [字段] [*] [函数操作] from 表名 where 条件 group by 分组条件 having 过滤条件 order by
```
-- 执行顺序
```sql
from ..where...group by ... having...order by select
- 注意点2: select + 分组条件 + 其他操作 其中的 分组条件只能有一个
- 注意点3: where 和 having 和区别
– where 后面不能接 聚合函数
– having在group by 后面执行 可以接聚合函数