一、视图

视图是一个虚拟表(非真实存在的),其本质是‘根据SQL语句获取动态的数据集,并为其命名‘ ,用户使用时只需使用“名称”即可获取结果集,可以将该结果集当做表来使用。

使用视图我们可以把查询过程中的临时表摘出来,用视图去实现,这样以后再想操作该临时表的数据时就无需重写复杂的SQL了,直接去视图中查找即可,但视图有明显的效率问题,并且视图是存放在数据库中的,如果我们程序中使用的SQL过分依赖数据库中的视图,即强耦合,那就意味着扩展SQL极为不便,因此并不推荐使用

视图有如下的特点:

    1.视图的列可以来自不同的表,是表的抽象和逻辑意义上建立的新关系
    2.视图是有基本表(实表)产生的表(虚表)
    3.视图的建立和删除不影响基本表
    4.对视图内容的更新(添加、删除和修改)直接影响基本表
    5.当视图来自多个基本表时,不允许添加,修改和删除数据

 1.创建视图

 

语法:
create view 视图名称 as sql语句
ps:create view db1 as select name from ren;

 

#!!!注意注意注意:

#1. 使用视图以后就无需每次都重写子查询的sql,但是这么效率并不高,还不如我们写子查询的效率高

 

#2. 而且有一个致命的问题:视图是存放到数据库里的,如果我们程序中的sql过分依赖于数据库中存放的视图,那么意味着,一旦sql需要修改且涉及到视图的部分,则必须去数据库中进行修改,而通常在公司中数据库有专门的DBA负责,你要想完成修改,必须付出大量的沟通成本DBA可能才会帮你完成修改,极其地不方便

 2 使用视图

 

语法
select * from 视图名称;

ps:修改视图,原始表也会跟着改
伪需求

1)查询表
select * from course

2)创建一个course的视图
create view course_view as select *from course;

3)查询course_view视图
select *from course_vies 

4)更新course_view视图的内容
update course_view set cname='xxxx';

5)往视图中插入数据
insert into course_view values(5,'yyy',2);

6)查看原始表,发现原始表的记录也跟着修改了
select * from course; 

 

 我们不应该修改视图中的记录,而且在涉及多个表的情况下是根本无法修改视图中的记录的

3.修改视图

语法:

alter view 视图名称 as sql语句

ps:alter view teacher_view as select * from course where cid>3;

4.删除视图

语法:
drop view 视图名称
ps:DROP VIEW teacher_view

二、触发器

使用触发器可以定制用户对表进行【增、删、改】操作时前后的行为,无查询  

触发器:监视某种情况,并触发某种操作
触发器创建语法四要素:

    1.监视地点(table)
    2.监视事件(insert/update/delete)
    3.触发时间(after/before)
    4.触发事件(insert/uodate/delete)

1.创建触发器语法

create trigger 触发器名称 after/before insert/update/delete
    on 表名 for each row
begin

#需要执行的sql语句

end

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

注意1:after/before:只能选一个,after 表示 后置触发,before 表示前置触发
注意2:insert/update/delete:只能选一个

需求:创建两张表

#商品表

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

);

问题

1.添加3条商品数据

insert into goods(name,num) values ('苹果','10'),('草莓','15'),('金桔','20'); 

如果我们在没使用触发器之前:假设我们现在卖了3个商品1,我们需要做两件事  

NO.1 往订单表插入一条记录

insert into ORDER_table(gid,much) VALUES (1,3) ;

NO.2 更新商品表苹果的剩余数量  

update goods set num = num - 3 WHERE name = '苹果';

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

NOW现在,我们来创建一个触发器:

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);

就会发现苹果的数量变成了7,说明在我们插入一条订单的时候,

触发器自动帮我们做了更新操作。

但现在会有一个问题,因为我们触发器里面num和ID都写死了,所以不管我们买那个商品,最终更新的都是苹果的数量。比如:我们往订单表再插入一条记录:

insert into order_table(gid,much) values(2,3);

 执行完后会发现苹果的数量变4了,而草莓的数量没变,这样显然不是我们想要的结果。我们需要改改我们之前创建的触发器。

 

我们如何在触发器引用行的值,也就是说我们要得到我们新插入的订单记录中的gid或much的值。

对于insert而言,新插入的行用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,name) values(2,3);

执行完代码,发现草莓的num变成了7,现在就对了  

现在还存在两种情况:

1.当用户撤销一个订单的时候,我们这边直接删除一个订单,我们是不是需要把对应的商品数量再加回去呢?

 对于delete而言:原本有一行,后来被删除,想引用被删除的这一行,用old来表示旧表中的值,old.列名可以引用原(旧)表中的值。

那我们的触发器就该这样写

create trigger tg3 afert delete  on order_table
for each row
bigen
    update goods set num = num + old.much where id = old.gid;-- (注意这边的变化)
end

2.当用户修改一个订单的数量时,我们触发器修改怎么写?  

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

3.存储过程

Mysql数据库在5.0版本后开始之初存储过程,那么什么是存储过程呢?怎么创建、查看和删除存储过程呢?存储过程有什么优点?

存储过程:类似于函数(方法),简单的说存储是为了完成某个数据库中特定功能而编写的语句集合,该语句集包括SQL语句(对数据的增删改查)、条件语句和循环语句等。

 1. 查看现有的存储过程  

show procedure status;

2 .删除存储过程   

drop procedure 存储过程名称;

3. 调用 存储过程  

call 存储过程名称(参数入/出类型 参数名 数据类型);

 4.创建存储过程  

1.体会封装
#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 account where id = i;
end
 
-- 调用
set @name =null;
CALL p2(1,@name);
select @name;

 

注意1: mysql中有三种出入参数类型:分别为:1. in 入参类型  2.out 出参类型   3. inout 出入参类型 

注意2: into 关键字 可以 将前面字段的查询结果 执行 给 into 后面的变量.

3.SQL 体会控制 

#3.SQL 体会控制
 create procedure p3(in x int,in c char(1))
 begin
    if c ='d' then
         select * from account where money >x;
   else
         select * from account where money <x;     
  end if;
end

 4.体会循环:计算1-100累加的和,并且返回计算结果. 

#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、不便于系统后期维护

4.函数 

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
MySQL内建函数

相关文章:

  • 2021-12-20
  • 2022-12-23
  • 2021-10-15
  • 2021-05-11
  • 2022-12-23
猜你喜欢
  • 2021-08-09
  • 2021-07-01
  • 2022-01-29
  • 2022-12-23
  • 2022-12-23
相关资源
相似解决方案