Mysql学习---视图/触发器/存储过程/函数/执行计划/sql优化 180101

时间:2022-09-12 20:35:02

视图

视图: 视图是一个虚拟表(非真实存在),动态获取数据,仅仅能做查询操作

本质:【根据SQL语句获取动态的数据集,并为其命名】,用户使用时只需使用【名称】即可获取结果集,并可以将其当作表来使用。由于视图是虚拟表,所以无法使用其对真实表进行创建、更新和删除操作,PyMysql是支持视图的。

仅能做查询用。

创建视图:
create VIEW stu as select * from student; # 这里只是建立了一个对应关系,视图是虚表,动态获取数据
select * from stu; # 这里只是简化了操作,实际上还是执行了select * from student
查看视图:
show TABLES # 会显示table和view视图信息
删除视图:
drop VIEW stu;
修改视图:
ALTER VIEW stu as select * from student where gender = '男';
PyMysql是支持视图的

Mysql学习---视图/触发器/存储过程/函数/执行计划/sql优化 180101

触发器

对某个表进行【增/删/改】操作的前后如果希望触发某个特定的行为时,可以使用触发器

触发器用于定制用户对表的行进行【增/删/改】前后的行为

触发器无法由用户直接调用,而知由于对表的【增/删/改】操作被动引发的。

特别的:

NEW表示即将插入的数据行,OLD表示即将删除的数据行

多行操作的时候,每一行都会进行一个轮询操作

触发器的范围: INSERT、DELETE、UPDATE

触发器的时机: BEFORE、AFTER

创建触发器: 特别的:NEW表示即将插入的数据行,OLD表示即将删除的数据行

CREATE TRIGGER tri_before_insert_tb1 BEFORE INSERT ON student FOR EACH ROW
BEGIN
IF NEW.gender == 'M' THEN # NEW == student
INSERT into Male_stu(sex) VALUES('M'); # 输入性别为M,则插入Male_stu
ELSE
INSERT into Feamle_stu(sex) VALUES('W');
ELSEIF
END
INSERT into student(gender, class_id, sname) values('W', 1, '哈哈哈')
注意: 更新操作需要2个值,一个NEW传入的值,一个OLD的值

Mysql学习---视图/触发器/存储过程/函数/执行计划/sql优化 180101

删除触发器

drop TRIGGER tri_before_insert_tb1

存储过程

存储过程是一个SQL语句集合[可增删改查在一个函数里],当主动去调用存储过程时,其中内部的SQL语句会按照逻辑执行;内部可又有for等语句。

注意:执行存储过程,肯定会先执行里面的sql语句的,且只能返回一条结果集,所有有多表的联合查询操作是,最好合并为一条结果集返回。

存储过程

a. 可写复杂逻辑

b. 参数:in   out   inout

c. 结果集:select ...

# 创建无参数的存储过程,类似函数的创建

CREATE PROCEDURE p1()
BEGIN
select * from student;
END

# 存储过程调用

call p1()          # 使用CALL 存储名即可, 执行存储过程,显示结果

删除存储过程:

drop procedure p1;
 

# 创建有参数的存储过程,用来执行自定义变量和获取sql集[结果集只能有一个,但可以拼接结果集]

# 对于存储过程,可以接收参数,其参数有三类:
                in         仅用于传入参数用
             out         仅用于返回值用,在内部直接赋值后外部调用可以直接获取到内容[Mysql自动帮我们建立链接关系]
           inout        既可以传入又可以当作返回值

create procedure p3(in i1 int,in i2 int,inout i3 int, out r1 int)
BEGIN
DECLARE temp1 int; # DECLARE声明变量,且存储过程里面必须使用
DECLARE temp2 int default 0; # 声明默认变量值
set temp1 = 1;
set r1 = i1 + i2 + temp1 + temp2;
set i3 = i3 + 100; # 功能一:自定义的函数操作
select * from student; # 功能二:查询并返回结果集,且一次只能返回一个,但可以拼接结果集
END; # 注意封号是用来执行结果的,没有封号则不会在执行存储
-- 执行存储过程: 使用CALL 存储名即可
set @t1 =4; # 必须带@符号
set @t2 = 0;
CALL p3 (1, 2 ,@t1, @t2); # 执行存储,并且自动返回了select * 的结果结合
SELECT @t1,@t2; # 单独执行此行,仅仅返回了自定义的函数结果

Mysql学习---视图/触发器/存储过程/函数/执行计划/sql优化 180101

Mysql学习---使用Python执行存储过程

事务:Innodb支持事务

事务用于将某些操作的多个SQL作为原子性操作,一旦有某一个出现错误,即可回滚到原来的状态,从而保证数据库数据完整性。

delimiter \ \
create
PROCEDURE
p1(
OUT
p_return_code
tinyint
)
BEGIN
DECLARE
exit
handler
for sqlexception
BEGIN
-- ERROR
set
p_return_code = 1;
rollback;
END; DECLARE
exit
handler
for sqlwarning
BEGIN
-- WARNING
set
p_return_code = 2;
rollback;
END; START
TRANSACTION;
DELETE
from tb1; insert
into
tb2(name)
values('seven');
COMMIT; -- SUCCESS
set
p_return_code = 0; END\ \
delimiter; 支持事务的存储过程

函数

函数: 内置函数 +  自定义函数

内置函数:

SELECT 1;
select CHAR_LENGTH('hello world'); #11,返回字符长度
SELECT CONCAT('hello ','world ','2017'); # hello world 2017,字符拼接
SELECT CONCAT_WS('_','hello ','world ','2017'); #hello _world _2017,添加了分隔符的字符拼接
SELECT CONV('8', 10, 2); #1000, 进制转换 10进制转化2进制
SELECT FORMAT(123456.2,2);#123,456.20 小数点后保留2位
SELECT LOWER('HELLO'); # 大写变小写
SELECT UPPER('hello'); # 小写变大写
SELECT INSERT('hello',0,2,'YY'); # 不更改,所以说明字符的替换是从第一个位置开始的
SELECT INSERT('hello',1,2,'YY'); # YYllo,从第一个位置开始替换
# 特别的:
# 如果pos超过原字符串长度,则返回原字符串
# 如果len超过原字符串长度,则由新字符串完全替换
SELECT INSTR('hello','e'); # 2, 返回e出现的索引位置
SELECT LEFT('hello', 3); #hel, 获取前3个字符
SELECT RIGHT('hello',3); #llo, 从右边取出3个值
SELECT SUBSTRING('hello',1,3); #hel, 默认从第一个位置开始取
SELECT TRIM(' ' ' hello ')# hello, 移除左右的空格
SELECT LTRIM(' hello world '); #helloworld,其引导空格字符被删除。
SELECT RTRIM(' hello world'); #hello world,结尾空格字符被删去
SELECT LOCATE('ll','hello'); # 3, 返回字符串所在的位置
SELECT REPEAT('h',5) #hhhhh, 重复前面的字符n次
SELECT REPLACE('hello','ll','yy')#heyyo, 替换字符
SELECT REVERSE('hello') #olleh, 字符反转
SELECT SPACE(2)# 返回2个空格

自定义函数:函数仅仅支持传递参数,返回一个结果,不允许写sql,不支持返回结果集

创建函数:
create function f1(i1 int,i2 int) # 传递2个参数
returns int # 返回结果,类似Java publist int f1(int i1, int i2)
BEGIN # 函数内容,函数内容不允许写sql, 不允许获取结果集
declare num int;
set num = i1 + i2;
# declare a int; # 函数里利用 select into 也可以实现赋值的操作
# select nid into a from student where name = 'hhh'; # 将nid值赋给a
return(num); # 返回结果
END;
执行函数:
SELECT f1(2,3) # 5
删除函数:
drop function f1;

函数和存储过程的区别:

Mysql学习---视图/触发器/存储过程/函数/执行计划/sql优化 180101

执行计划

相对比较准确表达出当前SQL运行状况,根据参考信息可以进行SQL优化一般显示All/Index的时候,效率不高,因为All 是全数据表扫描,index是全索引表扫描,而且rows里面的数据都是相对的,不是很准确。

- limit 的好处,找到第一个后就不在继续查找,效率相比较高

select * from tb1 where email='123'              -->[不推荐]

select * from tb1 where email='123' limit 1;   -->[推荐]

EXPLAIN select sid from student;

Mysql学习---视图/触发器/存储过程/函数/执行计划/sql优化 180101

EXPLAIN select sid from student;

Mysql学习---视图/触发器/存储过程/函数/执行计划/sql优化 180101

EXPLAIN select sid from student limit 1; # 也是从表扫描,但是找到第一条后,后面就不执行了

Mysql学习---视图/触发器/存储过程/函数/执行计划/sql优化 180101

EXPLAIN select sid from student where sid < 12;[所以创建表的时候,可以考虑将列设置为索引] 

Mysql学习---视图/触发器/存储过程/函数/执行计划/sql优化 180101

对SQL进行优化

- 对需要进行范围查找的列进行索引设置,因为在查找 <, <= 等进行操作的时候,使用的是Range范围查找,但是对于>, !=进行操作的时候,又是全局查找了

- 对于全表查找,最好加上limit, 因为有了limit查找到了数据后,就不在继续向下查找了

- 查询时的访问方式,性能:all < index < range < index_merge < ref_or_null < ref < eq_ref < system / const

- 避免使用select *

- count(1)或count(列) 代替 count(*)

- 创建表时尽量时 char 代替 varchar[char:定长用于固定长度的表单提交数据存储效率高,  varchar:不定长,效率偏低]

- 表的字段顺序固定长度的字段优先[varchar, text是不定长]

- 组合索引代替多个单列索引(经常使用多个条件查询时,组合索引比单独索引的合并快

- 尽量使用短索引[指定列的某几个字符为索引]

- 使用连接(JOIN)来代替子查询(Sub-Queries)

- 连表时注意条件类型需一致

- 索引散列值(重复少)不适合建索引,例:性别不适合

【更多参考】http://www.cnblogs.com/wupeiqi/articles/5713323.html   -->视图

【更多参考】http://www.cnblogs.com/wupeiqi/articles/5716963.html   -->索引

【更多参考】http://www.cnblogs.com/wupeiqi/articles/5716963.html   -->索引补充