mysql之视图,存储过程,触发器,事务

视图

视图是一个虚拟表(非真实存在),其本质是【根据SQL语句获取动态的数据集,并为其命名】,用户使用时只需使用【名称】即可获取结果集,可以将该结果集当做表来使用。

使用视图我们可以把查询过程中的临时表摘出来,用视图去实现,这样以后再想操作该临时表的数据时就无需重写复杂的sql了,直接去视图中查找即可,但视图有明显地效率问题,并且视图是存放在数据库中的,如果我们程序中使用的sql过分依赖数据库中的视图,即强耦合,那就意味着扩展sql极为不便,因此并不推荐使用。

1.创建视图

创建视图语法

CREATE VIEW 视图名称 AS SQL语句

准备数据和表

-- ========================
-- -- 创建部门表
create table dep(
id int primary key auto_increment,
name char(32)
);
-- 创建用户表
create table user(
id int primary key auto_increment,
name char(32),
dep_id int,
foreign key(dep_id) references dep(id)
);
-- 插数据
insert into dep(name) values('技术部'),('销售'),('财务部');
insert into user(name,dep_id) values ('egon',1),
                                       ('alex',2),
                                       ('jing',3);

 创建视图

create view user_dep_view as select dep_id,dep.name as dep_name, user.name as user_name from dep inner join user on user.dep_id=dep.id

 查询视图

 

-- 对于单表创建的视图来说是可以修改的,并且原来表的也就更改了。
create view dep_view as select * from dep where id=3;

select * from dep_view;


update dep_view set name='综合部' where id=3;
commit;
select * from dep;

insert into dep_view values(4,'人文部');
commit;

select * from dep;


delete from dep_view where id=3;
commit;
[Err] 1451 - Cannot delete or update a parent row: a foreign key constraint fails (`testmysql`.`user`, CONSTRAINT `user_ibfk_1` FOREIGN KEY (`dep_id`) REFERENCES `dep` (`id`))


-- 对于多表联合创建的视图是不可以修改的
insert into user_dep_view VALUES (5,'egon','人文部'); --会报错
[Err] 1394 - Can not insert into join view 'testmysql.user_dep_view' without fields list

DELETE from  user_dep_view where dep_id = 1;  --会报错
[Err] 1395 - Can not delete from join view 'testmysql.user_dep_view'

 

2.修改视图

语法:ALTER VIEW 视图名称 AS SQL语句

alter view dep_view as select * from dep where id=4;

select * from dep_view;

3.删除视图

语法:DROP VIEW 视图名称
drop view dep_view;
select * FROM dep_view;

  

触发器

使用触发器可以定制用户对表进行【增、删、改】操作时前后的行为,注意:没有查询

 

-- 触发器:某种程序触发了工具的运行
-- 触发器不能主动调用,只有触发了某种行为才会调用触发器的执行
-- 插入一条记录就触发一次

--创建语法
create
   trigger trigger_name
   trigger_time trigger_event
   on tbl_name for each row
   triggrr_body  #主体,就是在触发器里干什么事
trigger_time:{before | after}
trigger_event:{insert | update |detele}


# 插入前
CREATE TRIGGER tri_before_insert_tb1 BEFORE INSERT ON tb1 FOR EACH ROW
BEGIN
    ...
END

# 插入后
CREATE TRIGGER tri_after_insert_tb1 AFTER INSERT ON tb1 FOR EACH ROW
BEGIN
    ...
END

# 删除前
CREATE TRIGGER tri_before_delete_tb1 BEFORE DELETE ON tb1 FOR EACH ROW
BEGIN
    ...
END

# 删除后
CREATE TRIGGER tri_after_delete_tb1 AFTER DELETE ON tb1 FOR EACH ROW
BEGIN
    ...
END

# 更新前
CREATE TRIGGER tri_before_update_tb1 BEFORE UPDATE ON tb1 FOR EACH ROW
BEGIN
    ...
END

# 更新后
CREATE TRIGGER tri_after_update_tb1 AFTER UPDATE ON tb1 FOR EACH ROW
BEGIN
    ...
END

准备表

-- # 2.准备表
-- #第一步:准备表
create table cmd_log(
id int primary key auto_increment,
cmd_name char(64), #命令的名字
sub_time datetime, #提交时间
user_name char(32), #是哪个用户过来执行这个命令
is_success enum('yes','no')  #命令是否执行成功
);

create table err_log(
id int primary key auto_increment,
cname char(64), #命令的名字
stime datetime #提交时间
);

创建触发器

-- #创建触发器(向err_log表里插入最新的记录)
delimiter $$
create
  trigger tri_after_inser_cmd_log
  after insert
  on cmd_log for  each row
BEGIN
  if new.is_success = 'no' then
    insert into err_log(cname,stime) VALUES(new.cmd_name,new.sub_time);
  end if;  #记得加分号,mysql一加分号代表结束,那么就得声明一下
END $$
delimiter ;  #还原的最原始的状态


-- #创建触发器(向err_log表里插入最旧的记录)
delimiter $$
create
  trigger tri_after_del_cmd_log
  after delete
  on cmd_log for  each row
BEGIN
  if old.is_success = 'no' then
    insert into err_log(cname,stime) VALUES(old.cmd_name,old.sub_time);
  end if;  #记得加分号,mysql一加分号代表结束,那么就得声明一下
END $$
delimiter ;  #还原的最原始的状态  

语法小知识:

-- 触发器的两个关键字:new ,old
-- new :表示新的记录
-- old:表示旧的那条记录
-- 什么情况下才往里面插记录
-- 当命令输入错误的时候就把错误的记录插入到err_log表中

-- delimiter 详解
-- 其实就是告诉mysql解释器,该段命令是否已经结束了,mysql是否可以执行了。
-- 默认情况下,delimiter是分号;。在命令行客户端中,如果有一行命令以分号结束,那么回车后,mysql将会执行该命令。
DELIMITER $$ 
DROP TRIGGER IF EXISTS `updateegopriceondelete`$$ 
CREATE 
    TRIGGER `updateegopriceondelete` AFTER  DELETE ON  `customerinfo` 
    FOR EACH ROW BEGIN 
DELETE FROM egoprice  WHERE customerId=OLD.customerId; 
    END$$ 
DELIMITER 

-- 其中DELIMITER 定好结束符为"$$", 然后最后又定义为";", MYSQL的默认结束符为";". 
-- 详细解释: 
-- 其实就是告诉mysql解释器,该段命令是否已经结束了,mysql是否可以执行了。 
-- 默认情况下,delimiter是分号;。在命令行客户端中,如果有一行命令以分号结束, 
-- 那么回车后,mysql将会执行该命令。如输入下面的语句 
mysql> select * from test_table; 
-- 然后回车,那么MySQL将立即执行该语句。 
-- 但有时候,不希望MySQL这么做。在为可能输入较多的语句,且语句中包含有分号。

测试

# 测试
insert into cmd_log(cmd_name,sub_time,user_name,is_success) values
('ls -l /etc | grep *.conf',now(),'root','no'),
('ps aux |grep mysqld',now(),'root','yes'),
('cat /etc/passwd |grep root',now(),'root','yes'),
('netstat -tunalp |grep 3306',now(),'egon','no');
commit;


delete from cmd_log where is_success = 'yes';
delete from cmd_log where is_success = 'no';
commit;

select * from err_log;

删除触发器

drop trigger tri_after_insert_cmd;

 

事务

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

ACID,指数据库事务正确执行的四个基本要素的缩写。包含:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability)。一个支持事务(Transaction)的数据库,

必须要具有这四种特性,否则在事务过程(Transaction processing)当中无法保证数据的正确性,交易过程极可能达不到交易方的要求。

 

create table t_user(
id int primary key auto_increment,
name char(32),
balance int
);

insert into t_user(name,balance)
values
('yy',1000),
('xx',1000),
('zz',1000);

mysql> select * from t_user;
+----+------+---------+
| id | name | balance |
+----+------+---------+
|  1 | yy   |    1000 |
|  2 | xx   |    1000 |
|  3 | zz   |    1000 |
+----+------+---------+

#原子操作
start transaction;
update t_user set balance=900 where name='yy'; #买支付100元
update t_user set balance=1010 where name='xx'; #中介拿走10元
update t_user set balance=1090 where name='zz'; #卖家拿到90元
commit;


mysql> select * from t_user;
+----+------+---------+
| id | name | balance |
+----+------+---------+
|  1 | yy   |     900 |
|  2 | xx   |    1010 |
|  3 | zz   |    1090 |

#恢复表数据
truncate table t_user;
insert into t_user(name,balance)
values
('yy',1000),
('xx',1000),
('zz',1000);

mysql> select * from t_user;
+----+------+---------+
| id | name | balance |
+----+------+---------+
|  1 | yy   |    1000 |
|  2 | xx   |    1000 |
|  3 | zz   |    1000 |
+----+------+---------+

#出现异常,回滚到初始状态
start transaction;
update t_user set balance=900 where name='yy'; #买支付100元
update t_user set balance=1010 where name='xx'; #中介拿走10元
update t_user set balance=1090 where name='zz'; #卖家拿到90元,出现异常没有拿到
rollback;
commit;


mysql> select * from t_user;
+----+------+---------+
| id | name | balance |
+----+------+---------+
|  1 | yy   |    1000 |
|  2 | xx   |    1000 |
|  3 | zz   |    1000 |
+----+------+---------+  

在存储过程中使用事务

delimiter $$
create procedure p5(
	out p_return_code tinyint
    
)
begin 
declare exit handler for sqlexception
begin
	set p_return_code = 1;
    rollback;
end;

declare exit handler for sqlwarning
begin 
	set p_return_code = 2;
	rollback;
end;

start transaction ;
	DELETE FROM tb1; #执行失败
    insert into t_user values('zz',1000);
commit;
 -- SUCCESS 
    set p_return_code = 0; #0代表执行成功
    
end $$
delimiter ;  

 调用

#在mysql中调用存储过程
set @res=123;
call p5(@res);
select @res;

  

 

#在python中调用
# 有参数存储过程
    cursor.callproc('p5', args=('123',))  # 等价于cursor.execute("call p1()")

    # 获取执行完存储的参数,参数@开头
    cursor.execute("select @_p5_0;")  # @p2_0代表第一个参数,即返回值
    row_1 = cursor.fetchone()
    print(row_1) 
# Warning: (1146, "Table 'testmysql.tb1' doesn't exist")
#   self._do_get_result()
# {'@_p5_0': 1}

 

存储过程

一 存储过程介绍

存储过程包含了一系列可执行的sql语句,存储过程存放于MySQL中,通过调用它的名字可以执行其内部的一堆sql

使用存储过程的优点:

#1. 用于替代程序写的SQL语句,实现程序与sql解耦

#2. 基于网络传输,传别名的数据量小,而直接传sql数据量大

使用存储过程的缺点:

#1. 程序员扩展功能不方便

二 创建简单存储过程(无参)

delimiter $$
create procedure p6()
BEGIN

INSERT into test1(name,grade) values('egon4',100);
commit;
    
END $$
delimiter ;

#在mysql中调用存储过程
call p6() 

#在python中基于pymysql调用
cursor.callproc('p6') 
print(cursor.fetchall())

三 创建存储过程(有参)

对于存储过程,可以接收参数,其参数有三类:

#in          仅用于传入参数用
#out        仅用于返回值用
#inout     既可以传入又可以当作返回值
传入参数
create procedure p7(in_grade int)
BEGIN

select * from test1 where grade < in_grade;

    
END $$
delimiter ;

#在mysql中调用存储过程
call p7(100) ;

#在python中基于pymysql调用
cursor.callproc('p7',(100,))
print(cursor.fetchall())

 out:返回值

delimiter $$
create procedure p8(in in_grade int,out res int)
BEGIN

select * from test1 where grade < in_grade;
set res=1;

    
END $$
delimiter ;

#在mysql中调用存储过程
set @res=0; #0代表假(执行失败),1代表真(执行成功)
call p8(3,@res);
select @res;

# 在python中基于pymysql调用
    cursor.callproc('p8', (100, 0))  # 0相当于set @res=0
    print(cursor.fetchall())  # 查询select的查询结果

    cursor.execute('select @_p8_0,@_p8_1;')  # @p8_0代表第一个参数,@p8_1代表第二个参数,即返回值
    print(cursor.fetchall())

 inout:既可以传入又可以返回

delimiter $$
create procedure p9(inout inout_grade int)
BEGIN

select * from test1 where grade < inout_grade;
set inout_grade=60;

    
END $$
delimiter ;

#在mysql中调用存储过程
set @x=100; 
call p9(@x);
select @x;

 # 在python中基于pymysql调用
    cursor.callproc('p9', (100,))  # 0相当于set @res=0
    print(cursor.fetchall())  # 查询select的查询结果

    cursor.execute('select @_p9_0;')  # @p9_0代表第一个参数,即返回值
    print(cursor.fetchall())

四 执行存储过程

-- 无参数
call proc_name()

-- 有参数,全in
call proc_name(1,2)

-- 有参数,有in,out,inout
set @t1=0;
set @t2=3;
call proc_name(1,2,@t1,@t2)

 使用pymysql执行

#!/usr/bin/env python
# -*- coding:utf-8 -*-
import pymysql

conn = pymysql.connect(host='127.0.0.1', port=3306, user='root', passwd='123', db='t1')
cursor = conn.cursor(cursor=pymysql.cursors.DictCursor)
# 执行存储过程
cursor.callproc('p1', args=(1, 22, 3, 4))
# 获取执行完存储的参数
cursor.execute("select @_p1_0,@_p1_1,@_p1_2,@_p1_3")
result = cursor.fetchall()

conn.commit()
cursor.close()
conn.close()


print(result)

  

五 删除存储过程

drop procedure proc_name;

  




 

 

posted @ 2018-09-10 16:08  一只小小的寄居蟹  阅读(402)  评论(0编辑  收藏  举报