day040 数据库索引补充 存储过程 事务等
1.正确使用索引
视图: 关键词 view
视图是体格虚拟表
创建视图 : create view 视图名称 as sql语句;
例: create view t_view as select * from teacher where id =2;
使用视图: 如果修改了视图,那么原表也会更改
往视图中插入数据 : insert into t_view values (......);
修改视图 : alter view 视图名称 as sql语句
例: alter view v_view as select * from t2;
删除视图: drop view v_view;
触发器(trigger)
用户对某一个表进行操作(增删改查)的时候,触发某个操作(增删改查),成为触发器.
创建触发器: 插入前创建 : create trigger t_trigger before insert on s1 for each row 插入后创建 : create trigger t_trigger after insert on s1 for each row 删除前 : create trigger t_trigger before delete on s1 for each row 删除后: create trigger t_trigger after delete on s1 for each row 更新前: create trigger t_trigger before update on s1 for each row 更新后: create trigger t_trigger after update on for each row
插入后触发触发器:
实际中创建触发器
#准备表
CREATE TABLE cmd ( #这是一张指令信息表,你在系统里面执行的任何的系统命令都在表里面写一条记录
id INT PRIMARY KEY auto_increment, #id
USER CHAR (32), #用户
priv CHAR (10), #权限
cmd CHAR (64), #指令
sub_time datetime, #提交时间
success enum ('yes', 'no') #是否执行成功,0代表执行失败
); CREATE TABLE errlog ( #指令执行错误的信息统计表,专门提取上面cmd表的错误记录
id INT PRIMARY KEY auto_increment, #id
err_cmd CHAR (64), #错误指令
err_time datetime #错误命令的提交时间
);
#现在的需求是:不管正确或者错误的cmd,都需要往cmd表里面插入,然后,如果是错误的记录,还需要往
errlog表里面插入一条记录
#若果没有触发器,我们会怎么实现,我们完全可以通过咱们的应用程序来做,根据cmd表里面的success这个
字段是哪个值(yes成功,no表示失败),在给cmd插入记录的时候,判断一下这个值是yes或者no,来判断一
下成功或者失败,如果失败了,直接给errlog来插入一条记录
#但是mysql说,你的应用程序可以省事儿了,你只需要往cmd表里面插入数据就行了,没必要你自己来判断
了,可以使用触发器来实现,可以判断你插入的这条记录的success这个字段对应的值,然后自动来触发触发
器,进行errlog表的数据插入 #创建触发器
delimiter // (或者写$$,其他符号也行,但是不要写mysql不能认识的,知道一下就行了),
delimiter 是告诉mysql,遇到这句话的时候,就将sql语句的结束符分号改成delimiter后面的//
CREATE TRIGGER tri_after_insert_cmd AFTER INSERT ON cmd FOR EACH ROW
#在你cmd表插入一条记录之后触发的。
BEGIN #每次给cmd插入一条记录的时候,都会被mysql封装成一个对象,叫做NEW,里面的字段
都是这个NEW的属性
IF NEW.success = 'no' THEN #mysql里面是可以写这种判断的,等值判断只有一个等
号,然后写then
INSERT INTO errlog(err_cmd, err_time) VALUES(NEW.cmd, NEW.sub_time) ;
#必须加分号,并且注意,我们必须用delimiter来包裹,不然,mysql一看到分号,就认为你的sql
结束了,所以会报错
END IF ; #然后写end if,必须加分号
END// #只有遇到//这个完成的sql才算结束
delimiter ; #然后将mysql的结束符改回为分号 #往表cmd中插入记录,触发触发器,根据IF的条件决定是否插入错误日志
INSERT INTO cmd (
USER,
priv,
cmd,
sub_time,
success
)
VALUES
('chao','','ls -l /etc',NOW(),'yes'),
('chao','','cat /etc/passwd',NOW(),'no'),
('chao','','useradd xxx',NOW(),'no'),
('chao','','ps aux',NOW(),'yes'); #查询错误日志,发现有两条
mysql> select * from errlog;
+----+-----------------+---------------------+
| id | err_cmd | err_time |
+----+-----------------+---------------------+
| 1 | cat /etc/passwd | 2017-09-14 22:18:48 |
| 2 | useradd xxx | 2017-09-14 22:18:48 |
+----+-----------------+---------------------+
2 rows in set (0.00 sec)
删除触发器
drop frigger 触发器名;
事务
#原子操作
start transaction;
update user set balance=900 where name='wsb'; #买支付100元
update user set balance=1010 where name='chao'; #中介拿走10元
update user set balance=1090 where name='ysb'; #卖家拿到90元
commit; #只要不进行commit操作,就没有保存下来,没有刷到硬盘上 #出现异常,回滚到初始状态
start transaction;
update user set balance=900 where name='wsb'; #买支付100元
update user set balance=1010 where name='chao'; #中介拿走10元
uppdate user set balance=1090 where name='ysb'; #卖家拿到90元,出现异常没有拿到
rollback; #如果上面三个sql语句出现了异常,就直接rollback,数据就直接回到原来的状态了。但是执行了commit之后,rollback这个操作就没法回滚了
#我们要做的是检测这几个sql语句是否异常,没有异常直接commit,有异常就rollback,但是现在单纯的只是开启了事务,但是还没有说如何检测异常,我们先来一个存储过程来捕获异常,等我们学了存储过程,再细说存储过程。
commit;
事物的四大特性:
- 原子性
- 一致性
- 持久性
- 隔离性
事务用于将某些操作的多个SQL作为原子性操作,也就是这些sql语句要么同时成功,要么都不成功,事务的其他特性在我第一篇博客关于事务的介绍里面有,这里就不多做介绍啦,一旦有某一个出现错误,即可回滚到原来的状态,从而保证数据库数据完整性。
简单来说:我给一个姑娘转账,姑娘那儿收到了200,你的账户上扣了200,这两个操作是不是两个sql语句,这两个sql语句是你的应用程序发给mysql服务端的,并且这两个sql语句都要一起执行,不然数据就错了,你想想是不是。并且如果你通过应用程序发送这两条sql的时候,由于网络问题,你只发送了一个sql过来,那只有一个账户改了数据,另外一个没改,那数据是不是就出错了啊。这就是事务要完成的事情。
索引,创建的时候添加,添加索引的时候要注意,给字段里面数据大小比较小的字段添加,给字段里面的数据区分度高的字段添加
1) 索引未命中
如果我们在id字段加上了主键索引,只有select * from t1 where id=100;
where 后面的条件中没有用到id,那么就没有用到主键索引,这种情况就是索引未命中.
如果你有300w条数据,你的where 条件是id>100 ,我们会发现,随着你搜索的范围的增大,速度会越来越慢,会成倍的体现出来.
1.范围问题,或者说条件不明确,条件中出现了这些符号或者关键字' >, <, >= ,!= ,like ,between..and...'等
2.写索引的时候尽量选择区分度高的列作为索引,区分度公式是count(distinct col)/count(*),表示字段不重复的比例,比例越大我们扫描的记录越少,唯一键的区分度是1,而一些状态,性别字段可能在大数据面前区分度就是0.使用场景不同,这个值很难确定,一般需要join的字段我们都要求是0.1以上,即平均1秒扫描10条记录.
3.=和in可以乱序,比如a=1 and b=2 and c=3 建立a,b,c索引可以任意顺序,mysql的查询优化器会帮你优化成索引可以识别的形式.
4.索引列不能参与计算,比如:where id*10=10000;原因 ,b+树种存的都是数据表中的字段值,但进行检索时,需要把所有的元素都应用函数才能比较,显然成本太大,所以在写语句的时候,不要用索引进行 '' + - * / '' 操作 .
5.and 和 or
6.最左前缀匹配原则 ,对于组合索引mysql会一直向右匹配知道遇到范围查询(> < between like)就停止(指的是范围大了,有索引速度也慢)
普通索引 ,联合索引和唯一索引
普通索引:create index 索引名 on 表名(列名,)
drop index 索引名 on 表名
唯一索引: create unique index 索引名 on 表名(列名)
drop unique index 索引名 on 表名
联合索引: create unique index 索引名 on 表名(列名.列名)
drop unique index 索引名称 on 表名
建立所联合索引的一个原则: 索引是有个最左匹配的原则的,所以建联合索引的时候,将区分度高的放在最左边,依次排下来,范围查询的条件尽可能的往后面放.
联合索引的第二个好处是在第一个键相同的情况下,已经对第二个键进行了排序处理,例如再很多情况下应用程序都需要查询某个用户的购物情况,并按照时间进行排序,最后取出最近三次的购买记录,这是使用联合索引可以帮我们避免多一次的排序操作,因为索引本身在叶子节点已经排序了.
覆盖索引
innodb储存引擎支持覆盖引擎,即从辅助索引中就可以得到查询记录,二不需要查询聚集索引中记录.
唯一索引
create table t1(
id int unique,
); create table t1(
id int,
unique key unq_id(id)
); #表创建好之后添加唯一索引:
alter table t1 add unique key u_name(name); #删除
alter table t1 drop index u_name;
普通索引
index
#创建
create table t1(
id int,
index ind_id(id)
); alter table t1 add index ind_id(id); create index ind_id on t1(id); #删除
alter table t1 drop index ind_id(id);
drop index ind_id on t1;
存储过程
使用存储过程的优点:
1.用于代替程序写的sql语句,实现程序与sql解耦
2.基于网络传输,传别名的数据量小,而直接传sql数据量大
创建简单存储过程(无参)
delimiter //
create procedure p1()
begin
select * from blog;
insert into blog(name,sub_time) values('xxx',now()); end // delimiter ; call p1()
#类似于mysql的函数,但不是函数,mysql的函数都是放在mysql语句里面用的,不能单独的使用,存储过程是可以直接调用的 call 名字+() 调用;
#mysql 的视图啊 触发器啊等等都能在存储过程里面写,这是一大堆的sql的集合体,都可以综合到这里面 在python 中基于 pymysql 调用
cursor.callproc('p1')
print(cursor.fetchall())
储存过程中是可以传参数的
#对于储存过程,可以接收参数,参数有三种:
# in 仅用于传入参数
# out 仅用于返回值用
# inout 既可以用于传入也可以当做返回值
in 传入参数:
delimiter //
create procedure p2(
in n1 int, # n1参数是需要传入的,也就是接收外部数据,并且这个数据必须是int类型
in n2 int
)
begin select * from blog where id>n1 ;
end //
delimiter ; # 调用存储过程的两种方式,
call p2(1,2)
#在python中基于pymysql调用
cursor.callproc('p2',(1,2))
print(cursor.fetchall())
out 返回值:
#查看存储过程的一些信息:show create procedure p3; #查看视图啊、触发器啊都这么看,还可以
\用\G,show create procedure p3\G;\G的意思是你直接查看表结构可能横向上显示不完,\G是让表
给你竖向显示,一row是一行的字段
delimiter //
create procedure p3(
in n1 int,
out res int
)
BEGIN
select * from blog where id > n1;
set res = 1; #我在这里设置一个res=1,如果上面的所有sql语句全部正常执行了,那么这一句肯
定也就执行了,那么此时res=1,如果我最开始传入的时候,给res的值设置的是0,
#那么你想,最后我接收到的返回值如果是0,那么说明你中间肯定有一些sql语句执行失败了
#注意写法:out的那个参数,可以用set来设置,set设置之后表示这个res可以作为返回值,并且不需要像
python一样写一个return,你直接set之后的值,就是这个存储过程的返回值
END //
delimiter ; #在mysql中调用
set @res=0; #这是MySQL中定义变量名的固定写法(set @变量名=值),可以自己规定好,0代表假(执行
失败),1代表真(执行成功),如果这个被改为1了,说明存储过程中的sql语句执行成功了
call p3(3,@res);#注意:不要这样写:call p3(3,1),这样out的参数值你写死了,没法确定后面
这个1是不是成功了,也就是说随后这个out的值可能改成0了,也就是失败了,但是这样你就判断不了了,你
后面查看的这个res就成1了,所以这个参数应该是一个变量名昂,定义变量名就是上一句,如果你直接传一个
常量数字,会报错的,写法不对。
select @res; #看一下这个结果,就知道这些sql语句是不是执行成功了,大家明白了吗~~~ #在python中基于pymysql调用,在python中只需要知道存储过程的名字就行了
cursor.callproc('p3',(3,0)) #0相当于set @res=0,为什么这里这个out参数可以写常数0啊,因为
你用的pymysql,人家会帮你搞定,pymysql其实会帮你写成这样:第一个参数变量名:@_p3_0=3,第二
个:@_p3_1=0,也就是pymysql会自动帮你对应上一个变量名,pymysql只是想让你写的时候更方便
#沿着网络将存储过程名和参数发给了mysql服务端,比咱们发一堆的sql语句肯定要快对了,mysql帮你调
用存储过程
print(cursor.fetchall()) #查询select的查询结果 cursor.execute('select @_p3_0,@_p3_1;') #@_p3_0代表第一个参数,@_p3_1代表第二个参数,
即返回值
print(cursor.fetchall())
#别忘了关掉:
cursor.close()
conn.close()
删除存储过程
drop procedure proc_name;
day040 数据库索引补充 存储过程 事务等的更多相关文章
- mysql数据库----索引补充
1.索引 索引是表的目录,在查找内容之前可以先在目录中查找索引位置,以此快速定位查询数据.对于索引,会保存在额外的文件中. 2.索引种类 普通索引:仅加速查询 唯一索引:加速查询 + 列值唯一(可以有 ...
- 数据库表设计时一对一关系存在的必要性 数据库一对一、一对多、多对多设计 面试逻辑题3.31 sql server 查询某个表被哪些存储过程调用 DataTable根据字段去重 .Net Core Cors中间件解析 分析MySQL中哪些情况下数据库索引会失效
数据库表设计时一对一关系存在的必要性 2017年07月24日 10:01:07 阅读数:694 在表设计过程中,我无意中觉得一对一关系觉得好没道理,直接放到一张表中不就可以了吗?真是说,网上信息什么都 ...
- 代码中添加事务控制 VS(数据库存储过程+事务) 保证数据的完整性与一致性
做人事档案的系统考虑到数据的安全性与一致性,毕竟是要对外上线.真正投入使用的项目,数据库的可靠性与安全性上我们开发人员要考虑的就很多了,记得做机房收费系统时注册新卡是自己为了简单,写成了一个存储过程( ...
- SQL Server 数据库try catch 存储过程
SQL Server 在生产环境中这样写存储过程的坑都避免了吗? 原文链接: http://www.cnblogs.com/chenmh/p/7856777.html 概述 最近因为业务的需求写了一段 ...
- mysql数据库索引和引擎
1. 数据库索引 1.1 索引作用 当我们在数据库表中查询数据时,若没有索引,会逐个遍历表格中的所有记录,表格中数据记录量大时很耗时.建立索引就像创建目录一样,直接通过索引找到数据存储位置,加快查找. ...
- 好书推荐之Mysql三剑客 :《高性能Mysql》、《Mysql技术内幕》、《数据库索引设计与优化》
Mysql三剑客系列书籍: 大佬推荐 首先推荐<高性能 MySQL>,这本书是 MySQL 领域的经典之作,拥有广泛的影响力.不但适合数据库管理员(DBA)阅读,也适合开发人员参考学习.不 ...
- Java数据库连接——JDBC调用存储过程,事务管理和高级应用
一.JDBC常用的API深入详解及存储过程的调用 相关链接:Jdbc调用存储过程 1.存储过程(Stored Procedure)的介绍 我们常用的操作数据库语言SQL语句在执行的时候需要先编译,然后 ...
- mysql索引补充
索引补充 1.索引 索引是表的目录,在查找内容之前可以先在目录中查找索引位置,以此快速定位查询数据.对于索引,会保存在额外的文件中. 2.索引种类 普通索引:仅加速查询 唯一索引:加速查询 + 列 ...
- SQL Server 数据库索引
原文:SQL Server 数据库索引 一.什么是索引 减少磁盘I/O和逻辑读次数的最佳方法之一就是使用[索引] 索引允许SQL Server在表中查找数据而不需要扫描整个表. 1.1.索引的好处: ...
随机推荐
- MySQL学习(七)
学习子查询 1 查出本网站最新的good_id最大的一条商品(要求取出商品名) mysql> select goos_id,goods_name from goods -> order b ...
- Python Scrapy 爬虫框架实例(一)
之前有介绍 scrapy 的相关知识,但是没有介绍相关实例,在这里做个小例,供大家参考学习. 注:后续不强调python 版本,默认即为python3.x. 爬取目标 这里简单找一个图片网站,获取图片 ...
- Linux中ulimit -c生成core文件()
理解这六个shell脚本语言的功能 echo "kernel.core_pattern = /tmp/core-%e-%p-%t" >> /etc/sysctl.con ...
- (转)3个常用基于Linux系统命令行WEB网站浏览工具(w3m/Links/Lynx)
一般我们常用的浏览器肯定是基于可视化界面的图文结合的浏览界面效果,比如FireFox.Chrome.Opera等等,但是有些时候折腾和项目 的需要,在Linux环境中需要查看某个页面的文字字符,我们需 ...
- p1530 Fractions to Decimals
将余数记录下来,如果余数相同,那么商的下一位也相同. #include <iostream> #include <cstdio> #include <cmath> ...
- laravel调度任务
<?php namespace App\Console; use Illuminate\Console\Scheduling\Schedule;use Illuminate\Foundation ...
- Leetcode笔记(整理中.......)
1.树的层序遍历 使用两个List 数组 或者 两个queue队列,交替进行,将遍历到的一层放在第一个里面,之后再将遍历到的第二层放在第二个里面. 2.将一个无符号整数的二进制表示形式逆向输出 需要将 ...
- IDEA 错误:找不到或无法加载主类
下的java核心编程的源码,只有java文件,没有idea或者eclipse的项目结构信息. 分别用eclipse和idea打开了一遍,方便学习调试. 项目文件夹:E:\学习资料\Java\语法\ja ...
- spring cloud服务发现注解之@EnableDiscoveryClient与@EnableEurekaClient
使用服务发现的时候提到了两种注解,一种为@EnableDiscoveryClient,一种为@EnableEurekaClient,用法上基本一致,今天就来讲下两者,下文是从stackoverflow ...
- HBase环境搭建、shell操作及Java API编程
一. 1.掌握Hbase在Hadoop集群体系结构中发挥的作用和使过程. 2.掌握安装和配置HBase基本方法. 3.掌握HBase shell的常用命令. 4.使用HBase shell命令进行表的 ...