• 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','0755','ls -l /etc',NOW(),'yes'),
        ('chao','0755','cat /etc/passwd',NOW(),'no'),
        ('chao','0755','useradd xxx',NOW(),'no'),
        ('chao','0755','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 p3G;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;

  • 相关阅读:
    P4009 汽车加油行驶问题
    P2761 软件补丁问题
    P1251 餐巾计划问题
    P2766 最长不下降子序列问题
    P4011 孤岛营救问题
    P2765 魔术球问题
    P2770 航空路线问题
    P2762 太空飞行计划问题
    P2764 最小路径覆盖问题
    P3355 骑士共存问题
  • 原文地址:https://www.cnblogs.com/zty1304368100/p/10305747.html
Copyright © 2020-2023  润新知