zoukankan      html  css  js  c++  java
  • mysql高级

    一:视图

      1.为什么要有视图?

    select * from emp left join dep on emp.dep_id = dep.id
    union
    select * from emp right join dep on emp.dep_id = dep.id;
    
    create view temp(emp_id,emp_name,salary,dep_id,dep_id2,dep_name,work) as select * from emp left join dep on emp.dep_id = dep.id
    union
    select * from emp right join dep on emp.dep_id = dep.id;
    
    #全连接形成一张新表,现有查询是基于全连接这张新表,如何来操作/
    #解决:将新表起别名full_table,将需求转换为select需求字段from full_able 条件

      2.什么是视图?

    视图是由一张表或多张表的查询结果构成的一张虚拟表

      3.为什么用视图?

    将复杂常用的查询结果保留下来重复使用(将一张大表拆分成多张小表)

      4.语法

    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 reneme new_emp1;
    alter view new_emp modify id tinyint;
    
    
    视图中记录的操作:等价于普通表,完成增删改查
    update new_emp set 姓名='san' where id = 3;
    delete fro new_emp where id = 3;
    insert into new_emp(id, 姓名, 工资) values (10, "Bob", 10000);    #操作的是实体表,虚拟表要重新创建才能拿到最新数据
    
    视图的删除:
    drop view 视图名;
    
    总结:虚拟表的作用——查询

    二:触发器

      1.触发器是什么?

    当表数据发生更新时,会自动触发的功能称之为触发器

      2.为什么使用触发器?

    当一个表在数据发生更新时,需要去完成一些操作,可以为具体数据更新的方式添加触发器

    语法:
    delimiter //
    create trigger 触发器名 before | after insert | update | delete on 表名 for each row
    begin
    需要触发执行的sql代码
    end //
    delimiter;
    
    
    #触发器名:表名_(before|after)_(insert|update|delete)_trigger
    
    注:
    delimiter 是用来修改sql的语句结束标识符
    
    删除触发器:drop trigger 触发器名;
    # 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;

    三:事务

      1.事务是什么?

      事务是逻辑上的一组操作,要么都成功,要么都失败

      2.为什么使用事务?

      很多时候应该数据操作,不是应该sql语句就完成的,可能有很多个sql语句,如果部分sql执行成功而部分sql执行失败将导致数据错乱。(例如:转账=>转入转出均成功,才能认为操作成功)

       3.事务的使用

      start transaction ;——开启事务,在这条语句之后的sql将处在同一事务,并不会立即修改数据库

      commit ;——提交事务,让这个事务中的sql立即执行事务的操作

      rollback ;——回滚事务,取消这个事务,这个事务不会对数据库中的数据产生任何影响

      4.事务的四大特性:

      原子性:事务是一组不可分割的单位,要么同时成功,要么同时不成功

      一致性:事务前后的数据完整性应该保持一致(数据库的完整性:如果数据库在某一时间点下,所有的数据都符合所有的约束,则称数据库为完整性的状态)

      隔离性:事务的隔离性是指多个用户并发访问数据时,一个用户的事务不能被其他用户的事务所干扰,多个并发事务之间数据要相互隔离

      持久性:持久性是指一个事务一旦被提交,它对数据的改变就是永久性的,接下来几十数据库发生故障也不应该对其有影响

      5.事务的用户隔离级别

    数据库使用者可以控制数据库工作在哪个级别下,就可防止不同的隔离性问题

      read uncommitted ——不做任何隔离,可能脏读,幻读

      read committed ——可以防止脏读,不能防止不可重复读合幻读

      Repeatable read ——可以防止脏读,不可重复读,不能防止幻读

      Serializable——数据库运行在串行化实现,所有问题都没有,就是性能低

      6.修改隔离级别

      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()
    转账示例

    四:存储过程

      1.存储过程是什么?

      用于完成指定功能的sql语句块,类似于Python中的函数

      2.为什么?

      将能指定功能的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='数据库名';

    例子: 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;
    例子

    五:流程控制

    1.if语句的使用

    第一种 if:
    """
    if 条件 then
    语句;
    end if;
    """
    第二种 if elseif
    """
    if 条件  then
    语句1;
    elseif 条件 then
    语句2;
    else 语句3;
    end if;
    """

    案例:编写过程 实现 输入一个整数type 范围 1 - 2 输出 type=1 or type=2 or type=other;

    delimiter //
    create procedure showType(in type int,out result char(20))
    begin
    if type = 1 then 
    set result = "type = 1";
    elseif type = 2 then 
    set result = "type = 2";
    else 
    set result = "type = other";
    end if;
    end //
    delimiter ;
    
    set @res=null;
    call showType(100, @res);
    select @res;

    2.case语句

    大体意思与switch一样的(你给我一个值,我对它进行选择,然后执行匹配上的语句)

    语法:

    create procedure caseTest(in type int)
    begin
    CASE type 
    when 1  then select "type = 1";
    when 2  then select "type = 2";
    else select "type = other";
    end case;
    end

    定义变量:declare 变量名 类型 default 值;

    例如:declare i int default 0;

    3.while循环

    循环输出10次hello mysql
    create procedure showHello()
    begin 
    declare i int default 0;
    while  i < 10 do
    select "hello mysql";
    set i  = i + 1;
    end while;
    end

    4.loop循环

    没有条件,需要自己定义结束语句

    语法:

    输出十次hello mysql;
    create procedure showloop()
    begin 
    declare i int default 0;
    aloop: LOOP
    select "hello loop";
    set i = i + 1;
    if i > 9 then leave aloop;
    end if;
    end LOOP aloop;
    end

    5.repeat循环

    #类似do while
    #输出10次hello repeat
    create procedure showRepeat()
    begin
    declare i int default 0;
    repeat
    select "hello repeat";
    set i = i + 1;
    until i > 9
    end repeat;
    end
    
    #输出0-100之间的奇数
    create procedure showjishu()
    begin
    declare i int default 0;
    aloop: loop
    set i = i + 1;
    if i >= 101 then leave aloop; end if;
    if i % 2 = 0 then iterate aloop; end if;
    select i;
    end loop aloop;
    end
  • 相关阅读:
    转:PHP环境搭建
    证件号码
    3、SourceTree通过PUTTY连接GitLab
    2、gitlab 新建项目
    java基础31 List集合下的Vector集合
    java基础30 List集合下的LinkedList集合
    java基础29 迭代器 listIterator() 及各种遍历集合的方法
    java基础28 单例集合Collection下的List和Set集合
    java基础27 单例集合Collection及其常用方法
    错误/异常:The project cannot be built until build path errors are resolved 和 Unbound classpath container: 'JRE System Library [JavaSE-1.7]' in project 'MyJavaCode';的解决方法
  • 原文地址:https://www.cnblogs.com/liuxiaolu/p/10268708.html
Copyright © 2011-2022 走看看