MySql 进阶

1. 视图

1.1 视图的基本操作

视图(View)是一种虚拟存在的表。视图中的数据并不在数据库中实际存在,行和列数据来自定义视图的查询中使用的表,并且是在使用视图时动态生成的。

通俗的讲,视图只保存了查询的SQL逻辑,不保存查询结果。所以我们在创建视图的时候,主要的工作就落在创建这条SQL查询语句上

创建语法:

CREATE [OR REPLACE] VIEW 视图名称[(列名列表)] AS SELECT语句 [ WITH [ CASCADED | LOCAL ] CHECK OPTION ]
#创建视图 
CREATE OR REPLACE VIEW emp_v_1 AS SELECT * FROM emp WHERE sal <= 2000;

查询:

#查看创建视图语句:
SHOW CREATE VIEW 视图名称; 
#查看视图数据:
SELECT * FROM 视图名称 ...... ;
-- 查询视图 
SHOW CREATE VIEW emp_v_1; 
SELECT * FROM emp_v_1; 
SELECT * FROM emp_v_1 WHERE sal < 1000;

修改:

--方式一:
CREATE [OR REPLACE] VIEW 视图名称[(列名列表)] AS SELECT语句 [ WITH [ CASCADED | LOCAL ] CHECK OPTION ] 
​
--方式二:
ALTER VIEW 视图名称[(列名列表)] AS SELECT语句 [ WITH [ CASCADED | LOCAL ] CHECK OPTION ]
-- 修改视图 
CREATE OR REPLACE VIEW emp_v_1 AS SELECT empno,sal,ename FROM emp WHERE sal <= 2000;
​
ALTER VIEW emp_v_1 AS SELECT * FROM emp WHERE sal <= 2000;

删除:

DROP VIEW [IF EXISTS] 视图名称 [,视图名称] ...
drop view if exists emp_v_1;

上述我们演示了,视图应该如何创建、查询、修改、删除,那么我们能不能通过视图来插入、更新数据呢? 接下来,做一个测试

CREATE OR REPLACE VIEW emp_v_2 AS SELECT empno,ename,sal FROM emp_back WHERE sal<=2000;
​
SELECT * FROM emp_v_2;
​
INSERT INTO emp_v_2 VALUES(8080,'TOM',1888); 
​
INSERT INTO emp_v_2 VALUES(8081,'MARY',2500);

执行上述的SQL,我们会发现,empno为8080和8081的数据都是可以成功插入的。 但是我们执行查询,查询出来的数据,却没有id为8081的记录。

因为我们在创建视图的时候,指定的条件为 sal<=2000, id为8081的数据,是不符合条件的,所以没有查询出来,但是这条数据确实是已经成功的插入到了基表中。

如果我们定义视图时,如果指定了条件,然后我们在插入、修改、删除数据时,是否可以做到必须满足条件才能操作,否则不能够操作呢? 答案是可以的,这就需要借助于视图的检查选项了。

1.2 检查选项

当使用WITH CHECK OPTION子句创建视图时,MySQL会通过视图检查正在更改的每个行,例如 插入,更新,删除,以使其符合视图的定义。 MySQL允许基于另一个视图创建视图,它还会检查依赖视图中的规则以保持一致性。为了确定检查的范围,mysql提供了两个选项: CASCADED 和 LOCAL,默认值为 CASCADED 。

  • CASCADED 级联

    比如,v2视图是基于v1视图的,如果在v2视图创建的时候指定了检查选项为 cascaded,但是v1视图创建时未指定检查选项。 则在执行检查时,不仅会检查v2,还会级联检查v2的关联视图v1。

CREATE OR REPLACE VIEW emp_v_1 AS SELECT empno,ename,sal FROM emp_back WHERE sal<=2000;
​
​
CREATE OR REPLACE VIEW emp_v_2 AS SELECT  empno,ename,sal FROM emp_v_1 WHERE sal>=1000
WITH CASCADED CHECK OPTION;
​
INSERT INTO emp_v_2 VALUES(8080,'TOM',1888);
INSERT INTO emp_v_2 VALUES(8081,'MARY',2500);

执行上述的SQL,我们会发现,empno为8080的数据都是可以成功插入的。 而8081是无法插入的,这是因为, 2500 满足了emp_v_2的检查条件,但是emp_v_2视图是基于emp_v_1视图创建的,而是使用的是CASCADED,尽管emp_v_1没有设置with check option,但是emp_v_2使用with cascaded check option,所以在执行插入,不仅要检查插件的数据满足emp_v_2的条件,而且还有满足emp_v_1的条件,而8081满足emp_v_2条件,但是不满足emp_v_1,所以也插入失败,

  • LOCAL 本地

    比如,v2视图是基于v1视图的,如果在v2视图创建的时候指定了检查选项为 local ,但是v1视图创建时未指定检查选项。 则在执行检查时,知会检查v2,不会检查v2的关联视图v1。

CREATE OR REPLACE VIEW emp_v_1 AS SELECT empno,ename,sal FROM emp_back WHERE sal<=2000;
​
​
CREATE OR REPLACE VIEW emp_v_2 AS SELECT  empno,ename,sal FROM emp_v_1 WHERE sal>=1000
WITH LOCAL CHECK OPTION;
​
INSERT INTO emp_v_2 VALUES(8080,'TOM',1888);
INSERT INTO emp_v_2 VALUES(8081,'MARY',2500);

执行上述的SQL,我们会发现,empno为8080和8081的数据都是可以成功插入的。

1.3 视图的更新

要使视图可更新,视图中的行与基础表中的行之间必须存在一对一的关系。如果视图包含以下任何一项,则该视图不可更新:

A. 聚合函数或窗口函数(SUM()、 MIN()、 MAX()、 COUNT()等)

B. DISTINCT

C. GROUP BY

D. HAVING

E. UNION 或者 UNION ALL

create or replace view emp_v_1 as select job,count(1) from emp_back ;
INSERT INTO emp_v_1(job,sal) VALUES('市场',2000);

1.4 视图的作用

  • 简单

    视图不仅可以简化用户对数据的理解,也可以简化他们的操作。那些被经常使用的查询可以被定义为视图,从而使得用户不必为以后的操作每次指定全部的条件。

  • 安全

    数据库可以授权,但不能授权到数据库特定行和特定的列上。通过视图用户只能查询和修改他们所能见到的数据

  • 数据独立

    视图可帮助用户屏蔽真实表结构变化带来的影响。

案例

1). 为了保证数据库表的安全性,开发人员在操作tb_user表时,只能看到的用户的基本字段,屏蔽

手机号和邮箱两个字段

create view tb_user_view as select id,name,profession,age,gender,status,createtime from tb_user; 
​
select * from tb_user_view;

2). 查询每个学生所选修的课程(三张表联查),这个功能在很多的业务中都有使用到,为了简化操作,定义一个视图

create view tb_stu_course_view as select s.name student_name , s.no student_no , c.name course_name from student s, student_course sc , course c where s.id = sc.studentid and sc.courseid = c.id; 
​
select * from tb_stu_course_view;

2. 存储过程

2.1 存储过程简介

存储过程是事先经过编译并存储在数据库中的一段 SQL 语句的集合,调用存储过程可以简化应用开发人员的很多工作,减少数据在数据库和应用服务器之间的传输,对于提高数据处理的效率是有好处的。存储过程思想上很简单,就是数据库 SQL 语言层面的代码封装与重用。

优点:

  • 封装,复用 -----------------------> 可以把某一业务SQL封装在存储过程中,需要用到的时候直接调用即可。

  • 可以接收参数,也可以返回数据 --------> 再存储过程中,可以传递参数,也可以接收返回值。

  • 减少网络交互,效率提升 -------------> 如果涉及到多条SQL,每执行一次都是一次网络传输。 而如果封装在存储过程中,我们只需要网络交互一次可能就可以了。

2.2 基础语法

创建

CREATE PROCEDURE 存储过程名称 ([ IN/OUT/INOUT 参数名 参数类型 ]) 
BEGIN
    -- SQL语句 
END ;

调用:

CALL 名称 ([ 参数 ]); 

查看:

-- 查询指定数据库的存储过程及状态信息 
SELECT * FROM INFORMATION_SCHEMA.ROUTINES WHERE ROUTINE_SCHEMA = 'xxx'; 
​
-- 查询某个存储过程的定义
SHOW CREATE PROCEDURE 存储过程名称 ; 

删除:

DROP PROCEDURE [ IF EXISTS ] 存储过程名称;

注意:

在命令行中,执行创建存储过程的SQL时,需要通过关键字 delimiter 指定SQL语句的结束符。sqlyog 需要运行 执行所有查询选项

演示示例:

存储过程基本语法 :

-- 创建  
DELIMITER $$
CREATE PROCEDURE p1()
BEGIN
        SELECT COUNT(*) FROM emp;
END$$
DELIMITER ;
--调用
call p1();
-- 查看
select * from information_schema.ROUTINES where ROUTINE_SCHEMA = 'test';
​
show create procedure p1;
-- 删除 
drop procedure if exists p1;

2.3 变量

在MySQL中变量分为三种类型: 系统变量、用户定义变量、局部变量。

2.3.1 系统变量

系统变量 是MySQL服务器提供,不是用户定义的,属于服务器层面。分为全局变量(GLOBAL)、会话变量(SESSION)。

1.查看系统变量

#查看所有系统变量
SHOW [ SESSION | GLOBAL ] VARIABLES ;  
#可以通过LIKE模糊匹配方 式查找变量
SHOW [ SESSION | GLOBAL ] VARIABLES LIKE '......';  
#查看指定变量的值
SELECT @@[SESSION | GLOBAL] 系统变量名;

2.设置系统变量

SET [ SESSION | GLOBAL ] 系统变量名 = 值 ; 
SET @@[SESSION | GLOBAL] 系统变量名 = 值 ;

注意:

如果没有指定SESSION/GLOBAL,默认是SESSION,会话变量。

mysql服务重新启动之后,所设置的全局参数会失效,要想不失效,可以在 my.ini 中配置。

A. 全局变量(GLOBAL): 全局变量针对于所有的会话。

B. 会话变量(SESSION): 会话变量针对于单个会话,在另外一个会话窗口就不生效了。

-- 查看系统变量
show session variables ;
show session variables like 'auto%'; 
show global variables like 'auto%'; 
select @@global.autocommit; 
select @@session.autocommit; 
​
-- 设置系统变量 
set session autocommit = 1; 
insert into course(id, name) VALUES (6, 'ES'); 
set global autocommit = 0; 
select @@global.autocommit;
2.3.2 用户定义变量

用户定义变量 是用户根据需要自己定义的变量,用户变量不用提前声明,在用的时候直接用 "@变量名" 使用就可以。其作用域为当前连接。

  • 赋值

    --方式一
    SET @var_name = expr [, @var_name = expr] ... ; 
    SET @var_name := expr [, @var_name := expr] ... ;

    赋值时,可以使用 = ,也可以使用 := 。

    --方式二
    SELECT @var_name := expr [, @var_name := expr] ... ; 
    SELECT 字段名 INTO @var_name FROM 表名;

  • 使用

    SELECT @var_name ;

    注意: 用户定义的变量无需对其进行声明或初始化,只不过获取到的值为NULL。

  • 使用

    -- 赋值 
    set @myname = 'zhangsan'; 
    set @myage := 18; 
    set @mygender := '男',@myhobby := 'java'; 
    select @mycolor := 'red'; 
    select count(*) into @mycount from tb_user; 
    -- 使用 
    select @myname,@myage,@mygender,@myhobby; 
    select @mycolor , @mycount; 
    select @abc;
    2.3.3 局部变量

    局部变量 是根据需要定义的在局部生效的变量,访问之前,需要DECLARE声明。可用作存储过程内的局部变量和输入参数,局部变量的范围是在其内声明的BEGIN ... END块。

  • 声明

    DECLARE 变量名 变量类型 [DEFAULT ... ] ;

    变量类型就是数据库字段类型:INT、BIGINT、CHAR、VARCHAR、DATE、TIME等。

  • 赋值

    SET 变量名 = 值 ;
    SET 变量名 := 值 ; 
    SELECT 字段名 INTO 变量名 FROM 表名 ... ;
  • 演示示例:

    -- 声明局部变量 - declare 
    -- 赋值 
    create procedure p2() 
        begin
            declare stu_count int default 0; 
            select count(*) into stu_count from student; 
            select stu_count; 
        end;
        
     call p2();

3.索引

3.1.1 介绍

索引(index)是帮助MySQL高效获取数据的数据结构(有序)。在数据之外,数据库系统还维护着满足 特定查找算法的数据结构,这些数据结构以某种方式引用(指向)数据, 这样就可以在这些数据结构上实现高级查找算法,这种数据结构就是索引

一提到数据结构,大家都会有所担心,担心自己不能理解,跟不上节奏。不过在这里大家完全不用担 心,我们后面在讲解时,会详细介绍。

3.1.2 演示

表结构及其数据如下:

假如我们要执行的SQL语句为 : select * from user where age = 45;

1). 无索引情况

在无索引情况下,就需要从第一行开始扫描,一直扫描到最后一行,我们称之为 全表扫描,性能很低。

2). 有索引情况

如果我们针对于这张表建立了索引,假设索引结构就是二叉树,那么也就意味着,会对age这个字段建 立一个二叉树的索引结构。

此时我们在进行查询时,只需要扫描三次就可以找到数据了,极大的提高的查询的效率。

备注: 这里我们只是假设索引的结构是二叉树,介绍一下索引的大概原理,只是一个示意图,并不是索引的真实结构,索引的真实结构,后面会详细介绍。

3.1.3 特点

优势劣势
提高数据检索的效率,降低数据库的IO成本索引列也是要占用空间的。
通过索引列对数据进行排序,降低数据排序的成本,降低CPU的消耗。索引大大提高了查询效率,同时却也降低更新表的速度, 如对表进行INSERT、UPDATE、DELETE时,效率降低。

4.索引分类

在MySQL数据库,将索引的具体类型主要分为以下几类:主键索引、唯一索引、常规索引、全文索引。

分类含义特点关键字
主键索引针对于表中主键创建的索引默认自动创建, 只能有一个PRIMARY
唯一索引避免同一个表中某数据列中的值重复可以有多个UNIQUE
常规索引快速定位特定数据可以有多个
全文索引全文索引查找的是文本中的关键词,而不是比较索引中的值可以有多个FULLTEXT

聚集索引&二级索引

而在在InnoDB存储引擎中,根据索引的存储形式,又可以分为以下两种:

分类含义特点
聚集索引(Clustered Index)将数据存储与索引放到了一块,索引结构的叶子节点保存了行数据必须有,而且只有一个
二级索引(Secondary Index)将数据与索引分开存储,索引结构的叶子节点关联的是对应的主键可以存在多个

聚集索引选取规则:

如果存在主键,主键索引就是聚集索引。

如果不存在主键,将使用第一个唯一(UNIQUE)索引作为聚集索引。

如果表没有主键,或没有合适的唯一索引,则InnoDB会自动生成一个rowid作为隐藏的聚集索 引。

聚集索引和二级索引的具体结构如下:

聚集索引的叶子节点下挂的是这一行的数据 。

二级索引的叶子节点下挂的是该字段值对应的主键值。

接下来,我们来分析一下,当我们执行如下的SQL语句时,具体的查找过程是什么样子的

具体过程如下:

①. 由于是根据name字段进行查询,所以先根据name='Arm'到name字段的二级索引中进行匹配查找。但是在二级索引中只能查找到 Arm 对应的主键值 10。

②. 由于查询返回的数据是*,所以此时,还需要根据主键值10,到聚集索引中查找10对应的记录,最终找到10对应的行row。

③. 最终拿到这一行的数据,直接返回即可。

回表查询: 这种先到二级索引中查找数据,找到主键值,然后再到聚集索引中根据主键值,获取数据的方式,就称之为回表查询。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值