MySQL视图,函数,触发器,存储过程

Wesley13
• 阅读 526

1. 视图  

视图是一个虚拟表,它的本质是根据SQL语句获取动态的数据集,并为其命名,用户使用时只需使用【名称】即可获取结果集,可以将该结果集当做表来使用。
使用视图我们可以把查询过程中的临时表摘出来,用视图去实现,这样以后再想操作该临时表的数据时就无需重写复杂的sql了,直接去视图中查找即可,
但视图有明显地效率问题,并且视图是存放在数据库中的,如果我们程序中使用的sql过分依赖数据库中的视图,即强耦合,那就意味着扩展sql极为不便,
因此不推荐使用. 而且工作中一般不方便,因为是虚拟表 不方便共用,如果需要修改,可能设计到与DBA的沟通,很麻烦

 1  -- 使用视图
 2 select .. from v1
 3 select asd from v1
 4 -- 某个查询语句设置别名,日后方便使用
 5 
 6 - 创建
 7 create view 视图名称 as SQL
 8 
 9 PS: 虚拟的,临时表 无法插入操作 
10 
11 - 修改
12 alter view 视图名称 as SQL
13 
14 - 删除
15 drop view 视图名称;

2. 触发器

定制用户对表进行【增、删、改】操作时前后的行为,注意:没有查询
工作中一般也很少用到,因为自己在代码中就能设计操作前后的行为

insert into tb (....)

delimiter //   -- 修改结束标记
create trigger t1 BEFORE INSERT on student for EACH ROW   -- 创建insert操作前的触发器
BEGIN   -- 触发器具体内容
     INSERT into teacher(tname) values(NEW.sname);
     INSERT into teacher(tname) values(NEW.sname);
     INSERT into teacher(tname) values(NEW.sname);
     INSERT into teacher(tname) values(NEW.sname);
 END //  
delimiter ;  -- 恢复默认的语句结束标记

------------------------------------------------

insert into student(gender,class_id,sname) values('女',1,'涛'),('女',1,'根');

-- NEW,代指新数据 可以在触发器中点语法使用
-- OLD,代指老数据  删除的那一行记录被OLD引用

3.函数

因为在sql语句执行中调用函数会比较耗时,而且对索引的那一列使用了函数,则无法命中索引了。

所以工作中对响应速度要求高,一般不会不使用函数处理结果集。而是在架构级别或者程序级别处理结果集。

内置的函数很多,详情参看官方文档

-- 内置函数:
            执行函数 select CURDATE();
            
            blog
            id       title            ctime
             1        asdf        2019-11
             2        asdf        2019-11
             3        asdf        2019-10
             4        asdf        2019-10
             
             
            select ctime,count(1) from blog group ctime
            
            select DATE_FORMAT(ctime, "%Y-%m"),count(1) from blog group DATE_FORMAT(ctime, "%Y-%m")
            2019-11   2
            2019-10   2
            
        DATE_FORMAT  时间格式化函数,较常用
        
        
--        自定义函数(必须有返回值):
            
            delimiter \\
                create function f1(
                    i1 int,
                    i2 int)
                returns int
                BEGIN
                    declare num int default 0;
                    set num = i1 + i2;
                    return(num);
                END \\
            delimiter ;
            
            SELECT f1(1,100);

 4. 存储过程

包含了一系列可执行的sql语句,存储过程存放于MySQL中,通过调用它的名字可以执行其内部的一堆sql,可以让程序与sql解耦合,且执行通过一个名字减少数据传输。mysql 5.5版本以后才有的功能
开发岗位一般也比较少使用。主要会是DBA使用。

用MySQL的三种方式:

方式一:
  MySQL: 存储过程
  程序:调用存储过程
方式二:
  MySQL:。。
  程序:SQL语句
方式三:
  MySQL:。。
  程序:类和对象(SQL语句)

MySQL中代码属于强类型语言, 变量需要先 声明 变量名 和 变量类型.

MySQL视图,函数,触发器,存储过程 1.简单示例

MySQL视图,函数,触发器,存储过程 MySQL视图,函数,触发器,存储过程

delimiter //
create procedure p2(
    in n1 int,
    in n2 int
)
BEGIN
    ----- 获取大于传参数字的id行
    select * from student where sid > n1;
END //
delimiter ;

---------------- 命令行
call p2(12,2)

--------------- pymsql
cursor.callproc('p2',(12,2))

2.传参数(in)

MySQL视图,函数,触发器,存储过程 MySQL视图,函数,触发器,存储过程

delimiter //
create procedure p3(
    in n1 int,
    inout n2 int
)
BEGIN
    set n2 = 123123;
    select * from student where sid > n1;
END //
delimiter ;

------------------- 
set @v1 = 10;
call p2(12,@v1)
select @v1;

set @_p3_0 = 12
ser @_p3_1 = 2
call p3(@_p3_0,@_p3_1)
select @_p3_0,@_p3_1



------------------------ pymysql

cursor.callproc('p3',(12,2))
r1 = cursor.fetchall()
print(r1)


cursor.execute('select @_p3_0,@_p3_1')  #  @_p3_0 是底层创建好的名字
r2 = cursor.fetchall() # 去除out值
print(r2)

3.参数 out

为什么有结果集又有out伪造的返回值?

MySQL视图,函数,触发器,存储过程 MySQL视图,函数,触发器,存储过程

delimiter //
create procedure p3(
    in n1 int,
    out n2 int  -- 用于标识存储过程的执行结果  一般用 tinyint 1,2等来表示相应的执行结果,方便程序获取后知道执行结果
)
BEGIN
    insert into vv(..)
    insert into vv(..)
    insert into vv(..)
    insert into vv(..)
    insert into vv(..)
    insert into vv(..)
END //
delimiter ;

View Code

MySQL视图,函数,触发器,存储过程 MySQL视图,函数,触发器,存储过程

delimiter //
create procedure p4(
    out status int
)
BEGIN
    -- 伪代码描述
    1. 声明如果出现异常则执行{
        set status = 1;
        rollback;
    }
       
    开始事务
        -- 由秦兵账户减去100
        -- 方少伟账户加90
        -- 张根账户加10
        commit;
    结束
    
    set status = 2;
    
    
END //
delimiter ;

===============================
delimiter \\
create PROCEDURE p5(
    OUT p_return_code tinyint
)
BEGIN 
  DECLARE exit handler for sqlexception 
  BEGIN 
    -- ERROR 
    set p_return_code = 1; 
    rollback; 
  END; 
 
  START TRANSACTION; 
    DELETE from tb1;
    insert into tb2(name)values('seven');
  COMMIT; 
 
  -- SUCCESS 
  set p_return_code = 2; 
 
  END\\
delimiter ;

4.事务

MySQL视图,函数,触发器,存储过程 MySQL视图,函数,触发器,存储过程

delimiter //
create procedure p6()
begin 
    declare row_id int; -- 自定义变量1  
    declare row_num int; -- 自定义变量2 
    declare done INT DEFAULT FALSE;  -- 默认为false 表述循环未执行完
    declare temp int;
    
        -- 声明游标
    declare my_cursor CURSOR FOR select id,num from A;
    declare CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;
    
    -- 开始循环
    open my_cursor;
        xxoo: LOOP
            fetch my_cursor into row_id,row_num;
            if done then   --  需要自己判断是否循环结束
                leave xxoo; -- 结束循环
            END IF;
            set temp = row_id + row_num;
            insert into B(number) values(temp);
        end loop xxoo;
    close my_cursor;
    
end  //
delimter ;

5.游标-实现循环语句

游标性能比较差,一般很少用,使用场景是:针对每一行都需要专门的处理计算的时候可能会用到,但是一般update+ 循环也能解决 如:UPDATE B set num=id+num;

MySQL视图,函数,触发器,存储过程 MySQL视图,函数,触发器,存储过程

delimiter //
create procedure p7(
    in tpl varchar(255),
    in arg int
)
begin 
    set @xo = arg;
    PREPARE prod FROM 'select * from student where sid > ?'; -- 1. 预检测某个东西 SQL语句合法性
    EXECUTE prod USING @xo; -- 2. SQL =格式化 tpl + arg 
    DEALLOCATE prepare prod; -- 3. 执行SQL语句
end  //
delimter ;

---------------------------
call p7("select * from tb where id > ?",9)

6. 动态执行SQL(防SQL注入)

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Stella981 Stella981
3年前
Python3:sqlalchemy对mysql数据库操作,非sql语句
Python3:sqlalchemy对mysql数据库操作,非sql语句python3authorlizmdatetime2018020110:00:00coding:utf8'''
Easter79 Easter79
3年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
为什么mysql不推荐使用雪花ID作为主键
作者:毛辰飞背景在mysql中设计表的时候,mysql官方推荐不要使用uuid或者不连续不重复的雪花id(long形且唯一),而是推荐连续自增的主键id,官方的推荐是auto_increment,那么为什么不建议采用uuid,使用uuid究
Python进阶者 Python进阶者
10个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这