http://www.cnblogs.com/linhaifeng/articles/7495918.html

视图

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

'''
what: 视图是由一张表或多张表的查询结果构成的一张虚拟表
why: 将复杂常用的查询结果保留下来重复使用 | 将一张大表拆分成多张小表语法:
create [or replace] view 视图名[(查询字段别名们)] as 查询语句
create view new_emp as (select * from emp);注:
1.查询字段别名们 要与 查询语句的查询字段对应
2.create or replace: 操作视图没有则创建、有则替换
create or replace view new_emp(id,姓名,工资) as (select id,name,salary from emp where dep_id = 2);视图的修改:alter 等价于 create or replace, 且语法一致
alter view new_emp(id,姓名,工资) as (select id,name,salary from emp where dep_id = 1);视图中字段的操作:不允许alter操作字段
alter table new_emp rename new_emp1;
alter view new_emp modify id tinyint;视图中记录的操作:等价于普通表,完成增删改查
update new_emp set 姓名='san' where id = 3;
delete from new_emp where id = 3;
insert into new_emp(id, 姓名, 工资) values (10, "Bob", 10000); # 操作的是实体表, 虚拟表要重新创建才能拿到最新数据视图的删除:
drop view 视图名;总结: 虚拟表作用 -- 查询
'''

触发器

使用触发器可以定制用户对表进行【增删改】操作时前后的行为(不报包括查询)

'''
what:在表发生数据更新时,会自动触发的功能称之为触发器
why:当一个表在发生数据更新时,需要去完成一些操作,可以为具体数据更新的方式添加触发器语法:
delimiter //
create trigger 触发器名 before|after insert|update|delete on 表名 for each row
begin 需要触发执行的sql代码们
end //
delimiter ;# 触发器名: t1_before_insert_tri注:delimiter是用来修改sql的语句结束标识符删除触发器:drop trigger 触发器名;
'''
# 插入前
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# 删除触发器drop trigger tri_after_insert_cmd;
# cmd表
create table cmd (id int primary key auto_increment,user char(32),priv char(10),cmd char (64),sub_time datetime, # 提交时间success enum ('yes', 'no') # 0代表执行失败
);
# 错误日志表
create table errlog (id int primary key auto_increment,err_cmd char(64),err_time datetime
);
# 创建触发器
delimiter //
create trigger trigger1 after insert on cmd for each row
begin
# new就是cmd当前插入的那条记录(对象)
if new.success = "no" then # 等待值判断只有一个等号insert into errlog values(null, new.cmd, new.sub_time);
end if;
end //
delimiter ;# 往表cmd中插入记录,触发触发器,根据IF的条件决定是否插入错误日志
insert into cmd(user, priv, cmd, sub_time, success) values('egon', '0765', 'ls -l /etc', now(), 'yes'),('jerry', '0852', 'cat /etc/passwd', now(), 'no'),('kevin', '0867', 'useradd xxx', now(), 'no'),('owen', '0912', 'ps aux', now(), 'yes');
# 查看cmd数据信息
select * from cmd;
# 查看错误日志表中的记录是否有自动插入
select * from errlog;   

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

事务

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

'''
what:事务是逻辑上的一组操作,要么都成功,要么都失败
why:很多时候一个数据操作,不是一个sql语句就完成的,可能有很多个sql语句,如果部分sql执行成功而部分sql执行失败将导致数据错乱
eg:转账 => 转入转出均成功,才能认为操作成功事务的使用:
start transaction; --开启事物,在这条语句之后的sql将处在同一事务,并不会立即修改数据库
commit;--提交事务,让这个事物中的sql立即执行数据的操作,
rollback;--回滚事务,取消这个事物,这个事物不会对数据库中的数据产生任何影响事务的四大特性:
1.原子性:事务是一组不可分割的单位,要么同时成功,要么同时不成功
2.一致性:事物前后的数据完整性应该保持一致(数据库的完整性:如果数据库在某一时间点下,所有的数据都符合所有的约束,则称数据库为完整性的状态)
3.隔离性:事物的隔离性是指多个用户并发访问数据时,一个用户的事物不能被其它用户的事务所干扰,多个并发事务之间数据要相互隔离
4.持久性:持久性是指一个事物一旦被提交,它对数据的改变就是永久性的,接下来即使数据库发生故障也不应该对其有任何影响事务的用户隔离级别:
数据库使用者可以控制数据库工作在哪个级别下,就可与防止不同的隔离性问题
read uncommitted --不做任何隔离,可能脏读,幻读
read committed --可以防止脏读,不能防止不可重复读,和幻读,
Repeatable read --可以防止脏读,不可重复读,不能防止幻读
Serializable --数据库运行在串行化实现,所有问题都没有,就是性能低修改隔离级别:
select @@tx_isolation;--查询当前级别
set[session|global] transaction isolation level ....;修改级别
实例:
set global transaction isolation level Repeatable read;
注:修改后重新连接服务器生效
'''
#准备数据
create table account(id int primary key auto_increment,name varchar(20),money double
);
insert into account values(1,'owen',10000),(2,'egon',1000),(3,'jerry',1000),(4,'kevin',1000);# egon向owen借1000块钱
# 未使用事务
update account set money = money - 1000 where id = 1;
update account set moneys = money + 1000 where id = 2; # money打错了导致执行失败# 在python中使用事务处理
from pymysql.err import InternalError
sql = 'update account set money = money - 1000 where id = 1;'
sql2 = 'update account set moneys = money + 1000 where id = 2;' # money打错了导致执行失败
try:cursor.execute(sql)cursor.execute(sql2)conn.commit() # 提交事务
except InternalError:print("转账失败")conn.rollback()# 回滚操作

存储过程

一、简介

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

​ 优点:

#1. 用于替代程序写的SQL语句,实现程序与sql解耦#2. 基于网络传输,传别名的数据量小,而直接传sql数据量大

​ 缺点:程序员扩展功能不方便

程序与数据库结合使用的三种方式

#方式一:MySQL:存储过程程序:调用存储过程#方式二:MySQL:程序:纯SQL语句#方式三:MySQL:程序:类和对象,即ORM(本质还是纯SQL语句)

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

delimiter //
create procedure p1()
BEGINselect * from blog;INSERT into blog(name,sub_time) values("xxx",now());
END //
delimiter ;#在mysql中调用
call p1() #在python中基于pymysql调用
cursor.callproc('p1')
print(cursor.fetchall())

三、创建存储过程(有参)

对于存储过程,可以接收参数,其参数有三类:#in          仅用于传入参数用
#out        仅用于返回值用
#inout     既可以传入又可以当作返回值

① in:传入参数

delimiter //
create procedure p2(in n1 int,in n2 int
)
BEGINselect * from blog where id > n1;
END //
delimiter ;#在mysql中调用
call p2(3,2)#在python中基于pymysql调用
cursor.callproc('p2',(3,2))
print(cursor.fetchall())

② out:返回值

delimiter //
create procedure p3(in n1 int,out res int
)
BEGINselect * from blog where id > n1;set res = 1;
END //
delimiter ;#在mysql中调用
set @res=0; #0代表假(执行失败),1代表真(执行成功)
call p3(3,@res);
select @res;#在python中基于pymysql调用
cursor.callproc('p3',(3,0)) #0相当于set @res=0
print(cursor.fetchall()) #查询select的查询结果cursor.execute('select @_p3_0,@_p3_1;') #@p3_0代表第一个参数,@p3_1代表第二个参数,即返回值
print(cursor.fetchall())

③ inout 可传值可返回

delimiter //
create procedure p4(inout n1 int
)
BEGINselect * from blog where id > n1;set n1 = 1;
END //
delimiter ;#在mysql中调用
set @x=3;
call p4(@x);
select @x;#在python中基于pymysql调用
cursor.callproc('p4',(3,))
print(cursor.fetchall()) #查询select的查询结果cursor.execute('select @_p4_0;')
print(cursor.fetchall())

四、执行存储过程

① 在MySQL中执行存储过程

-- 无参数
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)

② 在python中基于MySQL执行存储过程

#!/usr/bin/env python
# -*- coding:utf-8 -*-
import pymysqlconn = 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;

六、小结

'''
what:用于完成指定功能的sql语句块,类似于Python中的函数
why:将能指定功能的sql语句块建立成存储过程,不仅将sql语句逻辑化了,更是功能化了,那我们要完成相同的事,只需要重复使用建立的存储过程,不就需要再重复书写sql语句了
# 总结: 存储过程可以让sql语句具有 复用性, 从而提高开发效率语法:
delimiter //
create procedure 存储过程名(输入输出类型1 参数名1 参数类型1(宽度), ... ,输入输出类型n 参数名n 参数类型n(宽度)
)
begin
sql语句块
end //
delimiter ;
注:
1.输入输出类型:in | out | inout
2.call 存储过程名(实参们)来调用存储过程案例:
set @res = null; # 定义空值变量, 用来接收存储过程的执行结果
delimiter //
create procedure user_info(in b int, in l int, out res char(20))
begin
select * from emp limit b, l;
set res = 'success';
end //
delimiter ;
call user_info(2, 3, @res); # 调用存储过程, 传入相应的实参
select @res; # 查看存储过程的执行结果变量的使用:
1.赋值变量:set @变量名 = 变量值
2.使用变量:@变量名 | select @变量名
3.删除变量:set @变量名 = null三种开发方式:
1. 业务逻辑 + 存储过程:高执行与开发效率,低耦合 | 不易移植,人员成本高
2. 业务逻辑 + 原生sql:人员成本低 | 开发难度大
3. 业务逻辑 + ORM:高开发效率,对象化操作数据库,可移植 | 性能消耗加大,多表联查、复杂条件会复制化ORM存储过程的操作:
1.查看
select routine_name, routine_type from information_schema.routines where routine_schema='数据库名';eg: select routine_name, routine_type from information_schema.routines where routine_schema='db2';2.删除
drop procedure [if exists] 数据库名.存储过程名
'''
delimiter //
create procedure send_money( out p_return_code char(20) )
begin # 异常处理declare exit handler for sqlexception begin # error set p_return_code = '错误异常'; rollback; end; # exit 也可以换成continue 表示发送异常时继续执行declare exit handler for sqlwarning begin # warning set p_return_code = '警告异常'; rollback; end; start transaction;update account set money = money - 1000 where id = 1;update account set money = moneys + 1000 where id = 2; # moneys字段导致异常commit; # success set p_return_code = '转账成功'; # 代表执行成功
end //
delimiter ;# 在mysql中调用存储过程
set @res=null;
call send_money(@res);
select @res;

转载于:https://www.cnblogs.com/prodigal/p/10263421.html

MySQL之视图、触发器、事务、存储过程相关推荐

  1. MySQL视图,触发器,事务,存储过程,函数

    create triggr triafterinsertcmdlog after insert on cmd_log FOR EACH ROW trigger_body .#NEW : 代表新的记录 ...

  2. mysql查看所有触发器以及存储过程等操作集合

    1. mysql查询所有表: SELECT TABLE_NAME FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA = '数据库名' AND  TAB ...

  3. mysql触发器_MySQL视图\触发器\事务初步认识

    视图 什么是视图? 视图就是通过查询得到虚拟表,然后将虚拟表保存下来,下次可以直接使用 为什么用视图? 如果要频繁的操作一张虚拟表,你就可以将该虚拟表制作成视图,后续可以直接进行操作 如何用视图 固定 ...

  4. MySQL-视图-触发器-事务-存储过程-函数-流程控制-索引与慢查询优化-06

    目录 视图*** 什么是视图 为什么要用视图 如何生成视图 修改视图 --> 最好(千万)不要 关联表数据改动前 关联表数据改动之后 触发器 什么是触发器 触发条件 触发器语法结构 修改mysq ...

  5. MySQL的视图、事务和索引

    视图 1. 为什么要有视图 对于复杂的查询,往往是有多个数据表进行关联查询而得到,如果数据库因为需求等原因发生了改变,为了保证查询出来的数据与之前相同,则需要在多个地方进行修改,维护起来非常麻烦 解决 ...

  6. mysql存储过程触发器_【MySQL笔记】触发器,存储过程和函数

    2)case语句 mysql > DELIMITER // mysql >CREATEPROCEDUREproc3 (inparameterint) ->begin ->dec ...

  7. MySQL 之视图、 触发器、事务、存储过程、内置函数、流程控制、索引(一)

    阅读目录 本文内容: 视图 触发器 事务 存储过程 内置函数 流程控制 索引 一.视图 视图就是通过查询得到一张虚拟表,然后保存下来,下次直接使用即可. 如果要频繁使用一张虚拟表,可以不用重复查询 视 ...

  8. mysql查询触发器_mysql查看所有触发器以及存储过程等操作集合【转】

    1. mysql查询所有表: SELECT TABLE_NAME FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA = '数据库名' AND  TAB ...

  9. day29 | 黑马程序员Java全程笔记 | 第二阶段MySQL高级事务-索引-视图-触发器-存储过程

    目录 01.反馈 02.回顾 03.并发访问MySQL-问题概述 并发访问的问题 04.并发访问MySQL-问题演示 05.并发访问MySQL-read-committed解决脏读问题 06.并发访问 ...

  10. MySQL 之 视图、触发器、存储过程、函数、事物与数据库锁

    浏览目录: 1.视图 2.触发器 3.存储过程 4.函数 5.事物 6.数据库锁 7.数据库备份 1.视图 视图:是一个虚拟表,其内容由查询定义.同真实的表一样,视图包含一系列带有名称的列和行数据 视 ...

最新文章

  1. python中time的用法_python中的计时器timeit的使用方法
  2. 虚拟机非法关机不能重启了
  3. java dbcp_Java dbcp连接池基本使用方法详解
  4. Python 经典模块可能都学过,10道题测试你会不会用?
  5. 暨南大学计算机复试线2019,暨南大学2019年考研复试分数线
  6. mariadb mysql表_mysql/mariadb学习记录——创建删除数据库、表的基本命令
  7. java getconstructor_java – 线程陷入getDeclaredConstructors的可能原因?
  8. @JVM内存模型(运行时数据区)
  9. Python - PyCharm部分快捷键
  10. 在线教育:最常见的直播课程有哪些?
  11. 彻底删除linux mbr,如何删除mbr分区表(超详细介绍Linux管理分区技巧)
  12. mysql getvar_杰奇登录后的东西都是在session里面的
  13. 解决SQLServer复制到Excel中内容换行问题
  14. 扬帆优配|太猛了!最高暴拉170%,港股这一板块狂飙!“带货”起飞?
  15. 入魔了~《武林外传》!
  16. Dubbox框架简介
  17. 开源5款工具漏洞扫描,打包带走吧!
  18. Google Earth谷歌地球卫片下载器--可下载全球历史卫星,无偏移、高精度
  19. Ubuntu上,tftpd-hpa 启动失败的解决方法.
  20. 电信黑莓手机出国漫游注意事项

热门文章

  1. SDK location not found. Define location with an ANDROID_SDK_ROOT environment variable or by setting
  2. 输入有序数组返回下标
  3. 二叉查找树的C语言实现(二)
  4. 适配器模式(为被封装对象提供不同的接口)
  5. 白话详细解读(二)-----AlexNet
  6. 第一个Java程序示例——Hello World
  7. Android adb无线调试脚本
  8. 彻底弄懂dalvik字节码【一】
  9. html怎么在页面中获取操作用户的id,CSS ID 选择器详解
  10. python自定义函数实例计算1-n的偶偶数和_python用户输入一个整数N,计算并输出1到N相加的和,请问这个程序错在哪里了?...