视图、触发器、存储过程、函数、事物、数据锁、数据库的备份 第四十一天 2018.12.04
视图(虚拟的表)view
在硬盘中一个表文件的存储
存储引擎决定一个文件分几个子文件存储
一般情况下存储引擎都用enging=innodb
innodb可以设置外键且支持锁
innodb下一个表(物理表)由两个子文件组成
.frm文件存储表结构不存储数据
.ibd文件是存储表的全部数据---->直接打开乱码
面试概念题
视图---->虚拟的表(只有结构没有数据)---->每次使用会根据字段到相应的表内寻找
视图和表的区别,视图没有实际的存储数据的文件
操作视图和操作表的方式一模一样
修改视图里的数据,相当于直接修改表内部的数据
视图的增删改查
创建视图(将某个SQL语句的查询结果创建为视图)
create view 视图名称 as sql 查询语句 #as为指向
删除视图
drop view 视图名称;
修改视图(更新视图)
alter view 视图名称 AS SQL语句
查看视图(使用视图)
select * from 视图名称;
视图有以下特点:
1. 视图的列可以来自不同的表,是表的抽象和逻辑意义上建立的新关系。
2. 视图是由基本表(实表)产生的表(虚表)。
3. 视图的建立和删除不影响基本表。
4. 对视图内容的更新(添加、删除和修改)直接影响基本表。
5. 当视图来自多个基本表时,不允许添加和删除数据。
单表可以增加、删除、修改
多表只能修改
触发器(增、删、改)trigger

触发器---->监视某种情况,并触发某种操作。
触发器创建语法四要素:1.监视地点(table)
2.监视事件(insert/update/delete)
3.触发时间(after/before)
4.触发事件(insert/update/delete)
删除触发器:drop trigger 触发器名;
查看触发器:show triggers
创建两张测试表
#商品表
create table goods(
id int primary key auto_increment,
name varchar(20),
num int
);
#订单表
create table order_table(
oid int primary key auto_increment,
gid int,
much int
);
#添加三条商品记录
insert into goods(name,num) values('商品1',10),('商品2',10),('商品3',10);
触发器语法:
create trigger triggerName after/before insert/update/delete on 表名 #创建触发器 触发器名称 在操作之前/之后 操作名称 在那个表上的操作
for each row #这句话是固定的 检查每行
begin
#需要执行的sql语句
end
注意1:after/before: 只能选一个 ,after 表示 后置触发, before 表示前置触发
注意2:insert/update/delete:只能选一个
没使用触发器之前:假设我们现在卖了3个商品1,我们需要做两件事
1.往订单表插入一条记录
insert into order_table(gid,much) values(1,3);
2.买完后,更新商品表商品1的剩余数量
update goods set num=num-3 where id=1;
创建一个触发器(固定参数)
create trigger tg1 after insert on order_table for each row begin update goods set num = num -3 where id = 1; end
只需要执行一条操作即可---->触发器自动帮我们做了更新操作。
insert into order_table(gid,much) values(1,3);
触发器实现传参(使用内置函数new来实现)
新插入的行用new来表示,行中的每一列的值用new.列名来表示
create trigger tg2 after insert on order_table for each row begin update goods set num = num-new.much where id = new.gid; end
使用完触发器后,不再使用触发器时需要删除掉
drop trigger tg1;
再次传入数据时就会得到需要的结果
insert into order_table(gid,much) values(2,3)
触发器撤销本次操作、触发器修改本次操作
当用户撤销一个订单的时候
create trigger tg3 afert delete on order_table
for each row
bigen
update goods set num = num + old.much where id = old.gid;-- (注意这边的变化)
end
当用户修改一个订单的数量时
create trigger tg4 after update on order_table
for each row
begin
update goods set num = num+old.much-new.much where id = old.gid;
end
存储过程(procedure)---->比较高级用法
MySQL数据库在5.0版本后开始支持存储过程
存储过程:类似于函数(方法),简单的说存储过程是为了完成某个数据库中的特定功能而编写的语句集合,
该语句集包括SQL语句(对数据的增删改查)、条件语句和循环语句等。
存储过程(查看、删除、调用)
查看现有的存储过程
show procedure status;
删除存储过程
drop procedure 存储过程名称;
调用存储过程
call 存储过程名称(参数入/出类型 参数名 数据类型);
创建存储过程
1、体会封装
create procedure p1 ()
begin
select * from account;
end
2、SQL体会传参
create procedure p2(in i int,out n varchar(50)) begin select name into n from goods where id = i; end -- 调用 set @name =null; CALL p2(1,@name); select @name;
MySQL的三种出入参数类型:in 入参类型 、out 出参类型 、inout 出入参类型
into 关键字 可以 将前面字段的查询结果 执行 给 into 后面的变量
传参数---->入参(in)、出参(out)、出入参(inout)
默认入参
in示例
-- 创建存储过程 in 入参
create procedure p_in (IN num int )
begin
select num;
set num=100;
select num;
end;
-- 调用
set @num=1;
call p_in(@num);
select @num;
-- 总结: IN 参数只是将变量在存储过程内部做了修改,并没有影响到外部,@num仍为1。
out示例
-- 创建存储过程 out 出参
create procedure p_out (out num int )
begin
select num;
set num=100;
select num;
end;
-- 调用
set @num=1;
call p_out(@num);
select @num;
inout示例
-- 创建存储过程 inout 出入参
create procedure p_inout (inout num int )
begin
select num;
set num=100;
select num;
end;
-- 调用
set @num=1;
call p_inout(@num);
select @num;
inout出入参示例
3、体会控制(if)
create procedure p3(in x int,in c char(5))
begin
if c ='true' then
select * from goods where money >x;
elseif c='false' then
select * from goods where money >x;
else
select * from goods where money =x;
end if;
end
4、体会循环(while do)
#4.体会循环:计算1-100累加的和,并且返回计算结果.
create procedure p4(inout n int)
begin
DECLARE sum int default 0; -- 设置总和变量,并且指定初始值0
declare i int; -- 声明变量
set i = 0; -- 通过set为变量设置值
while i<=n DO -- 开始循环
set sum = sum +i;
set i = i+1;
end while; -- 结束循环
select sum; -- 提供结果
set n = sum;--将计算结果提供给 输出变量 n;
end;
-- 调用:
set @n = 100;
call p4(@n);
select @n;
存储过程优点:
1、存储过程增强了SQL语言灵活性。
存储过程可以使用控制语句编写,可以完成复杂的判断和较复杂的运算,有很强的灵活性;
2、减少网络流量,降低了网络负载。
存储过程在服务器端创建成功后,只需要调用该存储过程即可,而传统的做法是每次都将大量的SQL语句通过网络发送至数据库服务器端然后再执行
3、存储过程只在创造时进行编译,以后每次执行存储过程都不需再重新编译。
一般SQL语句每执行一次就编译一次,所以使用存储过程可提高数据库执行速度。
存储过程缺点:
1、扩展功能不方便
2、不便于系统后期维护
函数(FUNCTION)
MySQL的内置函数:
一、数学函数
ROUND(x,y)
返回参数x的四舍五入的有y位小数的值
RAND()
返回0到1内的随机值,可以通过提供一个参数(种子)使RAND()随机数生成器生成一个指定的值。
二、聚合函数(常用于GROUP BY从句的SELECT查询中)
AVG(col)返回指定列的平均值
COUNT(col)返回指定列中非NULL值的个数
MIN(col)返回指定列的最小值
MAX(col)返回指定列的最大值
SUM(col)返回指定列的所有值之和
GROUP_CONCAT(col) 返回由属于一组的列值连接组合而成的结果
三、字符串函数
CHAR_LENGTH(str)
返回值为字符串str 的长度,长度的单位为字符。一个多字节字符算作一个单字符。
CONCAT(str1,str2,...)
字符串拼接
如有任何一个参数为NULL ,则返回值为 NULL。
CONCAT_WS(separator,str1,str2,...)
字符串拼接(自定义连接符)
CONCAT_WS()不会忽略任何空字符串。 (然而会忽略所有的 NULL)。
FORMAT(X,D)
将数字X 的格式写为'#,###,###.##',以四舍五入的方式保留小数点后 D 位, 并将结果以字符串的形式返回。若 D 为 0, 则返回结果不带有小数点,或不含小数部分。
例如:
SELECT FORMAT(12332.1,4); 结果为: '12,332.1000'
INSERT(str,pos,len,newstr)
在str的指定位置插入字符串
pos:要替换位置其实位置
len:替换的长度
newstr:新字符串
例如:
SELECT INSERT('abcd',1,2,'tt'); 结果为: 'ttcd'
SELECT INSERT('abcd',1,4,'tt'); 结果为: 'tt'
特别的:
如果pos超过原字符串长度,则返回原字符串
如果len超过原字符串长度,则由新字符串完全替换
INSTR(str,substr)
返回字符串 str 中子字符串的第一个出现位置。
LEFT(str,len)
返回字符串str 从开始的len位置的子序列字符。
例如:
SELECT INSTR('abc','c'); 结果为: 3
SELECT INSTR('abc','d'); 结果为: 0
LOWER(str)
变小写
UPPER(str)
变大写
REVERSE(str)
返回字符串 str ,顺序和字符顺序相反。
例如:
SELECT REVERSE('1234567') 结果为:7654321
SUBSTRING(str,pos) , SUBSTRING(str FROM pos) SUBSTRING(str,pos,len) , SUBSTRING(str FROM pos FOR len)
不带有len 参数的格式从字符串str返回一个子字符串,起始于位置 pos。带有len参数的格式从字符串str返回一个长度同len字符相同的子字符串,起始于位置 pos。
使用 FROM的格式为标准 SQL 语法。也可能对pos使用一个负值。假若这样,则子字符串的位置起始于字符串结尾的pos 字符,而不是字符串的开头位置。
在以下格式的函数中可以对pos 使用一个负值。
mysql> SELECT SUBSTRING('Quadratically',5); -- 从第5位开始截取
-> 'ratically'
mysql> SELECT SUBSTRING('foobarbar' FROM 4); -- 从第4位开始截取
-> 'barbar'
mysql> SELECT SUBSTRING('Quadratically',5,6); --从第5位开始截取,截取6个长度
-> 'ratica'
mysql> SELECT SUBSTRING('Sakila', -3); -- 从倒数第3位开始截取
-> 'ila'
mysql> SELECT SUBSTRING('Sakila', -5, 3); -- 从倒数第5位开始截取,截取3个长度
-> 'aki'
四、日期和时间函数
CURDATE()或CURRENT_DATE() 返回当前的日期
CURTIME()或CURRENT_TIME() 返回当前的时间
DAYOFWEEK(date) 返回date所代表的一星期中的第几天(1~7)
DAYOFMONTH(date) 返回date是一个月的第几天(1~31)
DAYOFYEAR(date) 返回date是一年的第几天(1~366)
DAYNAME(date) 返回date的星期名,如:SELECT DAYNAME(CURRENT_DATE);
FROM_UNIXTIME(ts,fmt) 根据指定的fmt格式,格式化UNIX时间戳ts
HOUR(time) 返回time的小时值(0~23)
MINUTE(time) 返回time的分钟值(0~59)
MONTH(date) 返回date的月份值(1~12)
MONTHNAME(date) 返回date的月份名,如:SELECT MONTHNAME(CURRENT_DATE);
NOW() 返回当前的日期和时间
QUARTER(date) 返回date在一年中的季度(1~4),如SELECT QUARTER(CURRENT_DATE);
WEEK(date) 返回日期date为一年中第几周(0~53)
YEAR(date) 返回日期date的年份(1000~9999)
重点:
DATE_FORMAT(date,format) 根据format字符串格式化date值
mysql> SELECT DATE_FORMAT('2009-10-04 22:23:00', '%W %M %Y');
-> 'Sunday October 2009'
mysql> SELECT DATE_FORMAT('2007-10-04 22:23:00', '%H:%i:%s');
-> '22:23:00'
mysql> SELECT DATE_FORMAT('1900-10-04 22:23:00',
-> '%D %y %a %d %m %b %j');
-> '4th 00 Thu 04 10 Oct 277'
mysql> SELECT DATE_FORMAT('1997-10-04 22:23:00',
-> '%H %k %I %r %T %S %w');
-> '22 22 10 10:23:00 PM 22:23:00 00 6'
mysql> SELECT DATE_FORMAT('1999-01-01', '%X %V');
-> '1998 52'
mysql> SELECT DATE_FORMAT('2006-06-00', '%d');
-> '00'
五、加密函数
MD5()
计算字符串str的MD5校验和
例如:
SELECT MD5('1234') 结果为:81dc9bdb52d04dc20036dbd8313ed055
PASSWORD(str)
返回字符串str的加密版本,这个加密过程是不可逆转的
例如:
SELECT PASSWORD('1234') 结果为:*A4B6157319038724E3560894F7F932C8886EBFCF
六、控制流函数
CASE WHEN[test1] THEN [result1]...ELSE [default] END
如果testN是真,则返回resultN,否则返回default
CASE [test] WHEN[val1] THEN [result]...ELSE [default]END
如果test和valN相等,则返回resultN,否则返回default
IF(test,t,f)
如果test是真,返回t;否则返回f
IFNULL(arg1,arg2)
如果arg1不是空,返回arg1,否则返回arg2
例如:
SELECT IFNULL('bbb','abc'); 结果为: bbb
SELECT IFNULL(null,'abc'); 结果为: abc
NULLIF(arg1,arg2)
如果arg1=arg2返回NULL;否则返回arg1
例如:
SELECT NULLIF('bbb','bbb');结果为: null
SELECT NULLIF('aaa','bbb');结果为: aaa
自定义函数
CREATE FUNCTION fun1(i1 int,i2 int)
RETURNS INT //设置返回类型
BEGIN
DECLARE sum int default 0;
set sum = i1+i2;
RETURN(sum); //返回结果
end
调用自定义函数
#直接调用自定义函数 select fun1(1,5); #在sql语句中使用自定义函数 select fun1(参数1,参数2),name from 表名
删除自定义函数
DROP FUNCTION fun_name;
事物(transaction)
事物---->几条SQL语句执行要么同时成功,要么同时失败
MySQL---->物理区、缓冲区(事物在缓冲区)
事物的4个特性(面试题)
原子性---->数据的修改,要么全成功要么全失败
一致性---->原来MySQL的规则,加上事物后规则不变
隔离性---->对同一个数据。只能同时有一个事物执行
持久性---->一旦提交了事物,就无法后悔了
事物的注意事项
- 在 MySQL 中只有使用了 Innodb 数据库引擎的数据库或表才支持事务。
- 事务处理可以用来维护数据库的完整性,保证成批的 SQL 语句要么全部执行,要么全部不执行。
- 事务用来管理 insert,update,delete 语句
事物的控制语句
-
BEGIN 或 START TRANSACTION;显式地开启一个事务;
-
COMMIT;也可以使用COMMIT WORK,不过二者是等价的。COMMIT会提交事务,并使已对数据库进行的所有修改称为永久性的;
-
ROLLBACK;有可以使用ROLLBACK WORK,不过二者是等价的。回滚会结束用户的事务,并撤销正在进行的所有未提交的修改;
- SAVEPOINT : 保存点,可以把一个事物分割成几部分.在执行ROLLBACK 时 可以指定在什么位置上进行回滚操作.
开启事物---->关闭MySQL的自动提交方式
commit---->提交当前事物、提交后也会清空缓冲区数据的执行结果
rollback---->回滚当前事物、从缓冲区清空掉数据执行结果 每次rollback只能回归上一个savepoint保存点
SET AUTOCOMMIT=0 ;禁止自动提交 和 SET AUTOCOMMIT=1 开启自动提交
例子
创建表
create table account(
id int(50) not null auto_increment primary key,
name VARCHAR(50) not null,
money DOUBLE(10,2) not NULL
);
#插入数据
insert into account (id,name,money) values(1,'鲁班',250),(2,'后羿',5000);
执行事物
start transaction; -- 开启事物 -- 执行sql语句操作 update account set money = money - 500 where id =1; update account set money = money+500 where id = 2; commit; -- 手动提交事物 rollback; -- 回滚事物 -- 查看结果 select * from account;
保存点使用
START TRANSACTION ;
insert into account (name,money) values('李元芳',1000);
SAVEPOINT s1; -- 设置保存点
insert into account (name,money) values('张桂枝',1500);
ROLLBACK to s1; -- 事物回滚到保存点 COMMIT; --提交事物
数据锁
锁的基本概念:当并发事务同时访问一个资源时,有可能导致数据不一致,因此需要一种机制来将数据访问顺序化,以保证数据库数据的一致性。
锁的基本类型: 共享锁、排它锁
例子
-- 窗口1 用户进行充值 -- 充值前 先查看余额 set @m=0; SELECT money into @m from account where id = 1; select @m; -- 看到余额后 充值100 块 update account set money = @m + 100 where id = 1; SELECT * from account; -------------------------------------------------------------- -- 窗口2 用户进行取款 -- 取款前 先查看余额 set @m=0; SELECT money into @m from account where id = 1; select @m; -- 看到余额后 取款100 块 update account set money = @m - 100 where id = 1; SELECT * from account;
共享锁(Shared Lock,也叫S锁)
共享锁(S)表示对数据进行读操作。因此多个事务可以同时为一个对象加共享锁。(如果试衣间的门还没被锁上,顾客都能够同时进去参观)
排他锁(Exclusive Lock,也叫X锁)
排他锁(X)表示对数据进行写操作。如果一个事务对 对象加了排他锁,其他事务就不能再给它加任何锁了。
(某个顾客把试衣间从里面反锁了,其他顾客想要使用这个试衣间,就只有等待锁从里面给打开了).
实际开发过程中遇到的两种锁:
悲观锁
就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block(阻塞)直到它拿到锁
悲观锁注意:要使用悲观锁,我们必须关闭mysql数据库的自动提交属性.因为MySQL默认使用autocommit模式,
也就是说,当你执行一个更新操作后,MySQL会立刻将结果进行提交。关闭自动提交命令为:set autocommit=0;
#开启多个界面测试
-- 0.开始事务 start transaction; -- 1.查询账户余额 set @m = 0; -- 账户余额 select money into @m from account where id = 1 for update; select @m; -- 2.修改账户余额 update account set money = @m -100 where id = 1; select * FROM account where id = 1; -- 3. 提交事务 commit;
会发现当前查询会进入到等待状态,不会显示出数据,当上面的sql执行完毕提交事物后,当前sql才会显示结果
注意1:在使用悲观锁时,如果表中没有指定主键,则会进行锁表操作
注意2: 悲观锁的确保了数据的安全性,在数据被操作的时候锁定数据不被访问,但是这样会带来很大的性能问题。因此悲观锁在实际开发中使用是相对比较少的。
乐观锁
每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据
使用乐观锁的两种方式
1.使用数据版本(Version)记录机制实现,这是乐观锁最常用的一种实现方式。
何谓数据版本?即为数据增加一个版本标识,一般是通过为数据库表增加一个数字类型的 “version” 字段来实现。
当读取数据时,将version字段的值一同读出,数据每更新一次,对此version值加一。
当我们提交更新的时候,判断数据库表对应记录 的当前版本信息与第一次取出来的version值进行比对。
-- 1.查询账户余额 set @m = 0; -- 账户余额 select money into @m from account where id = 1 ; select @m; -- 2.查询版本号 set @version = 0; -- 版本号 select version into @version from account where id = 1 ; select @version; -- 3.修改账户余额 update account set money = @m -100,version=version+1 where id = 1 and version = @version; select * FROM account where id = 1;
2.乐观锁定的第二种实现方式和第一种差不多,同样是在需要乐观锁控制的table中增加一个字段,名称无所谓
字段类型使用时间戳 (datatime), 和上面的version类似,也是在更新提交的时候检查当前数据库中数据的时间戳和自己更新前取到的时间戳进行对比
乐观锁适用于写入比较少的情况下,即冲突真的很少发生的时候,这样可以省去了锁的开销,加大了系统的整个吞吐量。
数据库备份
1.使用mysqldump实现逻辑备份(cmd端)
#语法: # mysqldump -h 服务器 -u用户名 -p密码 数据库名 > 备份文件.sql #示例: #单库备份 mysqldump -uroot -p123456 db1 > c:/db1.sql mysqldump -uroot -p123456 db1 table1 table2 > c:/db1-table1-table2.sql #多库备份 mysqldump -uroot -p123456 --databases db1 db2 mysql db3 > c:/db1_db2_mysql_db3.sql #备份所有库 mysqldump -uroot -p123456 --all-databases > c:/all.sql
2.恢复数据备份
#在mysql命令下,用source命令导入备份文件: mysql> USE 数据库名; mysql> source 备份文件.sql; 注意:只能在cmd界面下执行source命令,不能在mysql工具里面执行source命令,会报错,因为cmd是直接调用mysql.exe来执行命令的。
浙公网安备 33010602011771号