MySQL学习笔记(Day013:作业讲解一/Rank/视图/UNION/触发器)
@(MySQL学习)
[TOC]
一. 作业讲解
查询employees表中非基层用户的最近详细信息
老师的讲解的版本中存在问题,重新作为作业
统计dbt3库下orders每周每个客户的订单数量
思路
找到订单中最小周(week)之前的一周的周一,这里进行了简化,使用了
1970-01-05
作为周一标记,作为起始(start
)1
2
3
4
5
6
7
8[root@MyServer ~]> cal 1 1970
January 1970
Su Mo Tu We Th Fr Sa
1 2 3
4 5 6 7 8 9 10 # 1970-01-05 刚好是周一,用1月12号,19号,26号等也是可以的
11 12 13 14 15 16 17
18 19 20 21 22 23 24
25 26 27 28 29 30 31在起始条件周一(
start
)的基础上增加6天
时间,就是周日,即为一周的结束标记(end
)- 通过对
start
(周一)、end
(周日)以及o_custkey
进行分组
,使用count(o_orderkey)
得到对应的数量
SQL语句
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146-- 最终结果
select o_custkey,
ADDDATE('1970-01-05', INTERVAL FLOOR(DATEDIFF(o_orderdate, '1970-01-05')/7)*7 Day) as start,
ADDDATE('1970-01-05', INTERVAL FLOOR(DATEDIFF(o_orderdate, '1970-01-05')/7)*7 + 6 Day) as end,
count(o_orderkey) as total
from dbt3.orders group by o_custkey, start, end;
-- DATEDIFF(o_orderdate, '1970-01-05')
mysql> select datediff('1971-01-01', '1970-01-05'); -- 随意取一个1971-01-01,作为演示
+--------------------------------------+
| datediff('1971-01-01', '1970-01-05') |
+--------------------------------------+
| 361 | -- 1971-01-01 减去 1970-01-5 为361天
+--------------------------------------+
1 row in set (0.00 sec)
mysql> select datediff('1971-01-01', '1970-01-05') / 7; -- 求周数
+------------------------------------------+
| datediff('1971-01-01', '1970-01-05') / 7 |
+------------------------------------------+
| 51.5714 | -- 相隔51.5714周
+------------------------------------------+
1 row in set (0.00 sec)
-- FLOOR 和 ROUND函数
-- FLOOR(arg)是返回一个不大于arg的最大整数,其实就是取整
mysql> select floor(5.4);
+------------+
| floor(5.4) |
+------------+
| 5 |
+------------+
1 row in set (0.00 sec)
mysql> select floor(5.5);
+------------+
| floor(5.5) |
+------------+
| 5 |
+------------+
1 row in set (0.00 sec)
mysql> select floor(5.6);
+------------+
| floor(5.6) |
+------------+
| 5 |
+------------+
1 row in set (0.00 sec)
-- ROUND(X, D) 返回值是对数字X保留到小數点后D位,D默认为0,结果符合四舍五入原则;如果D为负数,则保留小数点左边(整数)的位数
mysql> select round(5.4); -- 默认D为0,四舍五入
+------------+
| round(5.4) |
+------------+
| 5 |
+------------+
1 row in set (0.00 sec)
mysql> select round(5.5); -- 默认D为0,四舍五入
+------------+
| round(5.5) |
+------------+
| 6 |
+------------+
1 row in set (0.00 sec)
mysql> select round(5.123, 1); -- 设D为1,保留小数点右边1为,四舍五入,不进位
+-----------------+
| round(5.123, 1) |
+-----------------+
| 5.1 |
+-----------------+
1 row in set (0.00 sec)
mysql> select round(5.163, 1); -- 设D为1,保留小数点右边1为,四舍五入,进位
+-----------------+
| round(5.163, 1) |
+-----------------+
| 5.2 |
+-----------------+
1 row in set (0.00 sec)
mysql> select round(524.163, -1); -- 保留小数点左边1位,不进位
+--------------------+
| round(524.163, -1) |
+--------------------+
| 520 |
+--------------------+
1 row in set (0.00 sec)
mysql> select round(524.163, -2); -- 保留小数点左边2位, 不进位
+--------------------+
| round(524.163, -2) |
+--------------------+
| 500 |
+--------------------+
1 row in set (0.00 sec)
mysql> select round(554.163, -2); -- 保留小数点左边2位, 进位
+--------------------+
| round(554.163, -2) |
+--------------------+
| 600 |
+--------------------+
1 row in set (0.00 sec)
-- 所以这里使用floor函数取整, 超过一周且不满一周的则落在start 和 end 中间
mysql> select floor(datediff('1971-01-01', '1970-01-05') / 7);
+-------------------------------------------------+
| floor(datediff('1971-01-01', '1970-01-05') / 7) |
+-------------------------------------------------+
| 51 | -- 取整为51周
+-------------------------------------------------+
1 row in set (0.00 sec)
mysql> select floor(datediff('1971-01-01', '1970-01-05') / 7) * 7;
+-----------------------------------------------------+
| floor(datediff('1971-01-01', '1970-01-05') / 7) * 7 |
+-----------------------------------------------------+
| 357 | -- 乘以7,得出357天差值
+-----------------------------------------------------+
1 row in set (0.00 sec)
mysql> select adddate('1970-01-05', interval floor(datediff('1971-01-01', '1970-01-05')/7)*7 day) as Monday_start;
-- 使用adddate函数,在1970-01-05(周一)的基础上,增加357天(51周),得到的值1970-12-28也是周一,标记为start
+--------------+
| Monday_start |
+--------------+
| 1970-12-28 | -- 1970-12-28(周一),而传入的值1971-01-01是该周的周五
+--------------+
1 row in set (0.00 sec)
-- 以同样的方法得到周日
mysql> select adddate('1970-01-05', interval floor(datediff('1971-01-01', '1970-01-05')/7)*7 + 6 day) as Sunday_end;
+------------+
| Sunday_end |
+------------+
| 1971-01-03 |
+------------+
1 row in set (0.00 sec)
-- 通过以上的周一和周日的计算,即可求出1971-01-01这天所在的一周,得出start(周一)和end(周日)
-- 然后对start和end以及o_custkey做分组操作,并通过count(o_orderkey)得到分组的定单量
- 使用子查询实现RowNumber
思路
- 假设当前在第N行记录,通过主键emp_no遍历有多少行的记录
小于等于
当前行,即为当前行的行数
- 假设当前在第N行记录,通过主键emp_no遍历有多少行的记录
SQL语句
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34SELECT
(SELECT COUNT(1) FROM employees b WHERE b.emp_no <= a.emp_no ) AS row_number,
emp_no,CONCAT(last_name," ",first_name) name,gender,hire_date
FROM employees a ORDER BY emp_no LIMIT 10;
-- 假设当前在第5行
mysql> select b.emp_no from employees.employees as b order by b.emp_no limit 5;
+--------+
| emp_no |
+--------+
| 10001 |
| 10002 |
| 10003 |
| 10004 |
| 10005 | -- 第5行的emp_no是10005
+--------+
5 rows in set (0.00 sec)
mysql> select count(*) from employees.employees as b where b.emp_no <= 10005 order by b.emp_no;
查找小于等于5的行数有几行
+----------+
| count(*) |
+----------+
| 5 | -- 小于等于10005的记录有5行,则5就是10005该行记录的行号
+----------+
1 row in set (0.00 sec)
-- 将该子查询的结果即可作为RowNumber
-- 但是该子查询的效率较低。不推荐使用。
-- 推荐Day012中提及的自定义变量方式
SELECT @a:=@a+1 AS row_number,
emp_no,CONCAT(last_name," ",first_name) name,gender,hire_date
FROM employees,(SELECT @a:=0) AS a LIMIT 10;
二. Rank
给出不同的用户的分数,然后根据分数计算排名
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59 mysql> create table test_rank(id int, score int);
Query OK, 0 rows affected (0.16 sec)
mysql> insert into test_rank values(1, 10), (2, 20), (3, 30), (4, 30), (5, 40), (6, 40);
Query OK, 6 rows affected (0.05 sec)
Records: 6 Duplicates: 0 Warnings: 0
mysql> select * from test_rank;
+------+-------+
| id | score |
+------+-------+
| 1 | 10 |
| 2 | 20 |
| 3 | 30 |
| 4 | 30 |
| 5 | 40 |
| 6 | 40 |
+------+-------+
6 rows in set (0.00 sec)
mysql> set @prev_value := NULL; -- 假设比较到第N行,设置一个变量prev_value用于存放第N-1行score的分数
-- 用于比较第N行的score和第N-1行的score
-- prev_value可以理解为 是临时保存第N-1行的score的变量
Query OK, 0 rows affected (0.00 sec)
mysql> set @rank_count := 0; -- 用于存放当前的排名
Query OK, 0 rows affected (0.00 sec)
mysql> select id, score,
-> case
-> when @prev_value = score then @rank_count
-- 相等则prev_value不变, 并返回rank_count(第一次为NULL,不会相等,所以跳转到下一个when语句)
-> when @prev_value := score then @rank_count := @rank_count + 1
-- 不等,则第N行的score赋值(:=)给prev_value。且rank_count增加1
-> end as rank_column -- case 开始的,end结尾
-> from test_rank
-> order by score desc;
+------+-------+-------------+
| id | score | rank_column |
+------+-------+-------------+
| 5 | 40 | 1 |
| 6 | 40 | 1 |
| 3 | 30 | 2 |
| 4 | 30 | 2 |
| 2 | 20 | 3 |
| 1 | 10 | 4 |
+------+-------+-------------+
6 rows in set (0.00 sec)
-- case
-- when [condition_1] then [do_something_1]
-- when [condition_2] then [do_something_2]
-- end
-- 语法:如果 condition_1条件满足,则执行 do_something_1 然后就跳出,不会执行condition_2;
-- 如果 condition_1条件不满足,则继续执行到 condition_2。以此类推。
--
-- 作业:使用一条语句写出Rank
--
三. 视图
- 创建视图
1 | -- |
视图的算法
视图的算法(ALGORITHM
)有三种方式:UNDEFINED
默认方式,由MySQL来判断使用下面的哪种算法MERGE
:每次
通过物理表
查询得到结果,把结果merge(合并)起来返回TEMPTABLE
: 产生一张临时表
,把数据放入临时表后,客户端再去临时表取数据(不会缓存
)TEMPTABLE 特点
:即使访问条件一样,第二次查询还是会去读取物理表中的内容,并重新生成一张临时表,并不会取缓存之前的表。(临时表是Memory存储引擎,默认放内存,超过配置大小放磁盘)当查询有一个较大的结果集时,使用
TEMPTABLE
可以快速的结束对该物理表的访问,从而可以快速释放这张物理表上占用的资源。然后客户端可以对临时表上的数据做一些耗时的操作,而不影响原来的物理表。
**所以一般我们使用`UNDEFINED`,由MySQL自己去判断**
四. UNION
UNION
的作用是将两个查询的结果集进行合并。- UNION必须由
两条或两条以上
的SELECT语句组成,语句之间用关键字UNION
分隔。 - UNION中的每个查询必须包含相同的列(
类型相同或可以隐式转换
)、表达式或聚集函数。
1 | mysql> create table test_union_1(a int, b int); |
如果知道数据本身具有唯一性,没有重复,则建议使用
union all
,因为union
会做去重操作
,性能会比union all
要低
五. 触发器
定义
- 触发器的对象是
表
,当表上出现特定的事件
时触发
该程序的执行
- 触发器的对象是
触发器的类型
UPDATE
- update 操作
DELETE
- delete 操作
- replace 操作
- 注意:drop,truncate等DDL操作
不会触发
DELETE
- 注意:drop,truncate等DDL操作
INSERT
- insert 操作
- load data 操作
- replace 操作
注意,
replace
操作会触发两次
,一次是UPDATE
类型的触发器,一次是INSERT
类型的触发器
MySQL 5.6版本同一个类型的触发器只能有一个(针对一个表)
MySQL 5.7允许多个同一类型的触发器注意:触发器只触发DML(Data Manipulation Language)操作,不会触发DDL(Data Definition Language)操作(create,drop等操作)
创建触发器
1
2
3
4
5
6
7
8
9
10
11
12
13CREATE
[DEFINER = { user | CURRENT_USER }]
TRIGGER trigger_name -- 触发器名字
trigger_time trigger_event -- 触发时间和事件
ON tbl_name FOR EACH ROW
[trigger_order]
trigger_body
trigger_time: { BEFORE | AFTER } -- 事件之前还是之后触发
trigger_event: { INSERT | UPDATE | DELETE } -- 三个类型
trigger_order: { FOLLOWS | PRECEDES } other_trigger_name
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
mysql> create table test_trigger_1 (
-> name varchar(10),
-> score int(10),
-> primary key (name));
Query OK, 0 rows affected (0.14 sec)
mysql> delimiter // -- 将语句分隔符定义成 // (原来是';')
mysql> create trigger trg_upd_score -- 定义触发器名字
-> before update on test_trigger_1 -- 作用在test_trigger_1 更新(update)之前(before)
-> for each row -- 每行
-> begin -- 开始定义
-> if new.score < 0 then -- 如果新值小于0
-> set new.score=0; -- 则设置成0
-> elseif new.score > 100 then -- 如果新值大于100
-> set new.score = 100; -- 则设置成100
-> end if; -- begin对应的 结束
-> end;// -- 结束,使用新定义的 '//' 结尾
Query OK, 0 rows affected (0.03 sec)
mysql> delimiter ; -- 恢复 ';' 结束符
-- new.col : 表示更新以后的值
-- old.col : 表示更新以前的值(只读)
mysql> insert into test_trigger_1 values ("tom", 200); -- 插入新值
Query OK, 1 row affected (0.04 sec)
mysql> select * from test_trigger_1;
+------+-------+
| name | score |
+------+-------+
| tom | 200 | -- 没改成100,因为定义的是update,而执行的是insert
+------+-------+
1 row in set (0.00 sec)
mysql> update test_trigger_1
-> set score=300 where name='tom'; -- 改成300
Query OK, 0 rows affected (0.00 sec)
Rows matched: 1 Changed: 0 Warnings: 0
mysql> select * from test_trigger_1;
+------+-------+
| name | score |
+------+-------+
| tom | 100 | -- 通过触发器的设置,大于100的值被修改成100
+------+-------+
1 row in set (0.00 sec)