视图

  • 视图
    • 概念:在mysql5之后出现的,是⼀种虚拟表,⾏和列的数据来⾃于定义视图时使⽤的⼀些表中,视图的数据是在使⽤视图的时候动态⽣成的,视图只保存sql的逻辑,不保存查询的结果。
    • 使⽤场景:多个地⽅使⽤到同样的查询结果,并且该查询结果⽐较复杂的时候,我们可以使⽤视图来隐藏复杂的实现细节
  • 视图的好处
    • 简化复杂的sql操作,不⽤知道他的实现细节
    • 隔离了原始表,可以不让使⽤视图的⼈接触原始的表,从⽽保护原始数据,提⾼了安全性
  • 创建视图 create view 视图名 as 查询语句;
  • 视图的使⽤步骤
    • 创建视图
    • 对视图执⾏查询操作
  • 更新视图【基本不⽤】
    • 视图的更新是更改视图中的数据,⽽不是更改视图中的sql逻辑
    • 当对视图进⾏更新后,也会对原始表的数据进⾏更新
    • 为了防⽌对原始表的数据产⽣更新,可以为视图添加只读权限,只允许读视图,不允许对视图进⾏更新
    • ⼀般情况下,极少对视图进⾏更新操作
  1. -- 查询姓名中包含a字符的员⼯名、部门、⼯种信息
  2. CREATE VIEW myv1
  3. AS
  4. SELECT
  5. t1.last_name,
  6. t2.department_name,
  7. t3.job_title
  8. FROM employees t1, departments t2, jobs t3
  9. WHERE t1.department_id = t2.department_id AND t1.job_id = t3.job_id;
  10. SELECT * FROM myv1 a where a.last_name like 'a%';
  11. -- 查询各部门的平均⼯资级
  12. CREATE VIEW myv2
  13. AS
  14. SELECT
  15. t1.department_id 部门id,
  16. t1.ag 平均⼯资,
  17. t2.grade_level ⼯资级别
  18. FROM (SELECT
  19. department_id,
  20. AVG(salary) ag
  21. FROM employees
  22. GROUP BY department_id)
  23. t1, job_grades t2
  24. WHERE t1.ag BETWEEN t2.lowest_sal AND t2.highest_sal;
  25. SELECT * FROM myv2;
  26. -- 修改视图
  27. -- ⽅式1:如果该视图存在,就修改,如果不存在,就创建新的视图
  28. CREATE OR REPLACE VIEW myv3
  29. AS
  30. SELECT
  31. job_id,
  32. AVG(salary) javg
  33. FROM employees
  34. GROUP BY job_id;
  35. -- ⽅式2alter 视图
  36. ALTER VIEW myv3
  37. AS
  38. SELECT *
  39. FROM employees;
  40. -- 删除视图 `drop view 视图名1 [,视图名2] [,视图名n];`
  41. drop view myv1,myv2,myv3;
  42. -- 查询视图结构
  43. -- 方式1desc
  44. desc 视图名称;
  45. -- 方式2show create view(显⽰了视图的创建语句)
  46. show create view 视图名称;

变量

  • 变量分类
    • 系统变量:系统变量由系统定义的,不是⽤户定义的,属于mysql服务器层⾯的
      • 全局变量:每次启动都会为所有的系统变量设置初始值;针对所有会话(连接)有效,可以跨连接,但不能跨重启,重启之后,mysql服务器会再次为所有系统变量赋初始值
      • 会话变量:针对当前会话(连接)有效,不能跨连接;会话变量是在连接创建时由mysql⾃动给当前会话设置的变量
    • ⾃定义变量:变量由⽤户⾃定义的,⽽不是系统提供的
      • 使用步骤:声明、赋值、使⽤(查看、⽐较、运算)
      • 分类:⽤户变量、局部变量
      • ⽤户变量:针对当前会话(连接)有效,作⽤域同会话变量;⽤户变量可以在任何地⽅使⽤也就是既可以在begin end⾥⾯使⽤,也可以在他外⾯使⽤
      • 局部变量:declare⽤于定义局部变量变量,在存储过程和函数中通过declare定义变量在begin…end中,且在语句之前。并且可以通过重复定义多个变量;作⽤范围同编程⾥⾯类似,在这⾥⼀般是在对应的begin和end之间。在end之后这个变量就没有作⽤了,不能使⽤了。
  • 赋值
    • 上⾯使⽤中介绍的,全局变量需要添加global关键字,会话变量需要添加session关键字,如果不写,默认为session级别
    • 全局变量的使⽤中⽤到了@@关键字,后⾯会介绍⾃定义变量,⾃定义变量中使⽤了⼀个@符号,这点需要和全局变量区分⼀下
  • delimiter关键字
    • 结束符,当mysql看到这个结束符的时候,表⽰可以执⾏前⾯的语句了,mysql默认以分号为结束符
    • 当我们创建存储过程或者⾃定义函数的时候,写了很⼤⼀⽚sql,⾥⾯包含了很多分号,整个创建语句是⼀个整体,需要⼀起执⾏,此时我们就不可⽤⽤分号作为结束符了
    • 那么我们可以通过delimiter关键字来⾃定义结束符
    • 默认结束符是分号
    • ⽤法:delimiter 分隔符
  1. -- 查看全局变量
  2. show global variables;
  3. -- 查看会话变量
  4. show session variables;
  5. show variables;
  6. -- 查看满⾜条件的系统变量量(like模糊匹配)
  7. show [global|session] like '%变量名%';
  8. -- 查看指定的系统变量(select和@@关键字,globalsession后⾯有个.符号)
  9. select @@[global.|session.]系统变量名称;
  10. -- 赋值⽅式1
  11. set [global|session] 系统变量名=值;
  12. -- 赋值⽅式2
  13. set @@[global.|session.]系统变量名=值;
  14. -- 用户变量
  15. -- 声明并初始化(要求声明时必须初始化)
  16. -- ⽅式1
  17. set @变量名=值;
  18. -- ⽅式2
  19. set @变量名:=值;
  20. -- ⽅式3
  21. select @变量名:=值;
  22. -- 赋值(更新变量的值)
  23. -- ⽅式1:这块和变量的声明⼀样
  24. set @变量名=值;
  25. set @变量名:=值;
  26. select @变量名:=值;
  27. -- ⽅式2
  28. select 字段 into @变量名 from 表;
  29. -- 使⽤
  30. select @变量名;
  31. -- 综合示例
  32. /*set⽅式创建变量并初始化*/
  33. set @username='路⼈甲java';
  34. /*select into⽅式创建变量*/
  35. select 'javacode2018' into @gzh;
  36. select count(*) into @empcount from employees;
  37. /*select :=⽅式创建变量*/
  38. select @first_name:='路⼈甲Java',@email:='javacode2018@163.com';
  39. /*使⽤变量*/
  40. insert into employees (first_name,email) values (@first_name,@email);
  41. -- 局部变量
  42. -- 声明
  43. declare 变量名 变量类型;
  44. declare 变量名 变量类型 [default 默认值];
  45. -- 赋值(注意:局部变量前⾯没有@符号)
  46. /*⽅式1*/
  47. set 局部变量名=值;
  48. set 局部变量名:=值;
  49. select 局部变量名:=值;
  50. /*⽅式2*/
  51. select 字段 into 局部变量名 from 表;
  52. -- 使⽤(查看变量的值)
  53. select 局部变量名;
  54. -- 示例
  55. /*创建表test1*/
  56. drop table IF EXISTS test1;
  57. create table test1(a int PRIMARY KEY,b int);
  58. /*声明脚本的结束符为$$*/
  59. DELIMITER $$
  60. DROP PROCEDURE IF EXISTS proc1;
  61. CREATE PROCEDURE proc1()
  62. BEGIN
  63. /*声明了⼀个局部变量*/
  64. DECLARE v_a int;
  65. select ifnull(max(a),0)+1 into v_a from test1;
  66. select @v_b:=v_a*2;
  67. insert into test1(a,b) select v_a,@v_b;
  68. end $$
  69. /*声明脚本的结束符为;*/
  70. DELIMITER ;
  71. /*调⽤存储过程*/
  72. call proc1();
  73. /*查看结果*/
  74. select * from test1;
  • 总结
    • 系统变量可以设置系统的⼀些配置信息,数据库重启之后会被还原
    • 会话变量可以设置当前会话的⼀些配置信息,对当前会话起效
    • declare创建的局部变量常⽤于存储过程和函数的创建中
    • 作⽤域:全局变量对整个系统有效、会话变量作⽤于当前会话、⽤户变量作⽤于当前会话、局部变量作⽤于begin end之间
    • 注意全局变量中⽤到了@@,⽤户变量变量⽤到了@,⽽局部变量没有这个符号
    • delimiter关键字⽤来声明脚本的结束符

存储过程&⾃定义函数详解

  • 需求背景介绍
    • 线上程序有时候出现问题导致数据错误的时候,如果⽐较紧急,我们可以写⼀个存储来快速修复这块的数据,然后再去修复程序
  • 存储过程
    • 概念:⼀组预编译好的sql语句集合,理解成批处理语句
    • 好处
      • 提⾼代码的重⽤性
      • 简化操作
      • 减少编译次数并且减少和数据库服务器连接的次数,提⾼了效率
  • 创建存储过程
    • 参数模式有3种:
    • in:该参数可以作为输⼊,也就是该参数需要调⽤⽅传⼊值
    • out:该参数可以作为输出,也就是说该参数可以作为返回值
    • inout:该参数既可以作为输⼊也可以作为输出,也就是说该参数需要在调⽤的时候传⼊值,又可以作为返回值
    • 参数模式默认为IN
    • ⼀个存储过程可以有多个输⼊、多个输出、多个输⼊输出参数
  1. -- 创建存储过程
  2. create procedure 存储过程名([参数模式] 参数名 参数类型)
  3. begin
  4. 存储过程体
  5. end
  6. -- 调⽤存储过程(关键字是call
  7. call 存储过程名称(参数列表);
  8. -- 删除存储过程(存储过程只能⼀个个删除,不能批量删除)
  9. drop procedure [if exists] 存储过程名称;
  10. -- 修改存储过程
  11. -- 存储过程不能修改,若涉及到修改的,可以先删除,然后重建
  12. -- 查看存储过程
  13. show create procedure 存储过程名称;
  14. -- 创建存储过程:空参列表
  15. /*设置结束符为$*/
  16. DELIMITER $
  17. /*如果存储过程存在则删除*/
  18. DROP PROCEDURE IF EXISTS proc1;
  19. /*创建存储过程proc1*/
  20. CREATE PROCEDURE proc1()
  21. BEGIN
  22. INSERT INTO t_user VALUES (1,30,'路⼈甲Java');
  23. INSERT INTO t_user VALUES (2,50,'刘德华');
  24. END $
  25. /*将结束符置为;*/
  26. DELIMITER ;
  27. /*调⽤存储过程*/
  28. CALL proc1();
  29. -- 创建存储过程:带in参数的存储过程
  30. /*设置结束符为$*/
  31. DELIMITER $
  32. /*如果存储过程存在则删除*/
  33. DROP PROCEDURE IF EXISTS proc2;
  34. /*创建存储过程proc2*/
  35. CREATE PROCEDURE proc2(id int,age int,in name varchar(16))
  36. BEGIN
  37. INSERT INTO t_user VALUES (id,age,name);
  38. END $
  39. /*将结束符置为;*/
  40. DELIMITER ;
  41. /*创建了3个⾃定义变量*/
  42. SELECT @id:=3,@age:=56,@name:='张学友';
  43. /*调⽤存储过程*/
  44. CALL proc2(@id,@age,@name);
  45. -- 创建存储过程:带out参数的存储过程
  46. delete a from t_user a where a.id = 4;
  47. /*如果存储过程存在则删除*/
  48. DROP PROCEDURE IF EXISTS proc3;
  49. /*设置结束符为$*/
  50. DELIMITER $
  51. /*创建存储过程proc3,前2个参数,没有指定参数模式,默认为in*/
  52. CREATE PROCEDURE proc3(id int,age int,in name varchar(16),out
  53. user_count int,out max_id INT)
  54. BEGIN
  55. INSERT INTO t_user VALUES (id,age,name);
  56. /*查询出t_user表的记录,放⼊user_count中,max_id⽤来存储t_user中最⼩的id*/
  57. SELECT COUNT(*),max(id) into user_count,max_id from t_user;
  58. END $
  59. /*将结束符置为;*/
  60. DELIMITER ;
  61. /*创建了3个⾃定义变量*/
  62. SELECT @id:=4,@age:=55,@name:='郭富城';
  63. /*调⽤存储过程*/
  64. CALL proc3(@id,@age,@name,@user_count,@max_id);
  65. -- 创建存储过程:带inout参数的存储过程
  66. /*如果存储过程存在则删除*/
  67. DROP PROCEDURE IF EXISTS proc4;
  68. /*设置结束符为$*/
  69. DELIMITER $
  70. /*创建存储过程proc4*/
  71. CREATE PROCEDURE proc4(INOUT a int,INOUT b int)
  72. BEGIN
  73. SET a = a*2;
  74. select b*2 into b;
  75. END $
  76. /*将结束符置为;*/
  77. DELIMITER ;
  78. /*创建了2个⾃定义变量*/
  79. set @a=10,@b:=20;
  80. /*调⽤存储过程*/
  81. CALL proc4(@a,@b);
  82. -- 查看存储过程
  83. show create procedure proc4;
  • 函数
    • 概念:⼀组预编译好的sql语句集合,理解成批处理语句,但是必须有返回值
  1. -- 创建函数(参数是可选的,返回值是必须的)
  2. create function 函数名(参数名称 参数类型)
  3. returns 返回值类型
  4. begin
  5. 函数体
  6. end
  7. -- 调⽤函数
  8. select 函数名(实参列表);
  9. -- 删除函数
  10. drop function [if exists] 函数名;
  11. -- 查看函数详细
  12. show create function 函数名;
  13. -- 创建函数:⽆参函数
  14. /*删除fun1*/
  15. DROP FUNCTION IF EXISTS fun1;
  16. /*设置结束符为$*/
  17. DELIMITER $
  18. /*创建函数*/
  19. CREATE FUNCTION fun1()
  20. returns INT
  21. BEGIN
  22. DECLARE max_id int DEFAULT 0;
  23. SELECT max(id) INTO max_id FROM t_user;
  24. return max_id;
  25. END $
  26. /*设置结束符为;*/
  27. DELIMITER ;
  28. /*调⽤效果*/
  29. SELECT fun1();
  30. -- 创建函数:有参函数
  31. /*删除函数*/
  32. DROP FUNCTION IF EXISTS get_user_id;
  33. /*设置结束符为$*/
  34. DELIMITER $
  35. /*创建函数*/
  36. CREATE FUNCTION get_user_id(v_name VARCHAR(16))
  37. returns INT
  38. BEGIN
  39. DECLARE r_id int;
  40. SELECT id INTO r_id FROM t_user WHERE name = v_name;
  41. return r_id;
  42. END $
  43. /*设置结束符为;*/
  44. DELIMITER ;
  45. /*调⽤效果*/
  46. SELECT get_user_id(name) from t_user;
  • 存储过程和函数的区别
    • 存储过程的关键字为procedure,返回值可以有多个,调⽤时⽤call,⼀般⽤于执⾏⽐较复杂的的过程体、更新、创建等语句
    • 函数的关键字为function,返回值必须有⼀个,调⽤⽤select,⼀般⽤于查询单个值并返回

流程控制语句介绍

  • if函数
  • case语句
  • if结构
  • while循环:类似于java中的while循环
  • repeat循环:类似于java中的do while循环
  • loop循环:类似于java中的while(true)死循环,需要在内部进⾏控制
  • 循环体控制语句
    • 结束本次循环:iterate 循环标签,类似于java中的continue
    • 退出循环:leave 循环标签,类似于java中的break
  1. -- if函数语法:if函数有3个参数,当参数1true的时候,返回值1,否则返回值2
  2. if(条件表达式,值1,值2);
  3. -- 实例如下:
  4. SELECT id 编号,if(sex=1,'男','⼥') 性别,name 姓名 FROM t_user;
  5. -- CASE结构语法
  6. case 表达式
  7. when 1 then 结果1或者语句1(如果是语句需要加分号)
  8. when 2 then 结果2或者语句2
  9. ...
  10. else 结果n或者语句n
  11. end [case] (如果是放在begin end之间需要加case,如果在select后则不需要)
  12. -- 示例1select中使⽤,查询t_user表数据,返回:编号、性别(男、⼥)、姓名
  13. SELECT id 编号,(CASE sex WHEN 1 then '男' WHEN 2 then '⼥' END)
  14. 别,name 姓名 FROM t_user;
  15. -- 示例2begin end中使⽤
  16. /*删除存储过程proc1*/
  17. DROP PROCEDURE IF EXISTS proc1;
  18. /*删除id=6的记录*/
  19. DELETE FROM t_user WHERE id=6;
  20. /*声明结束符为$*/
  21. DELIMITER $
  22. /*创建存储过程proc1*/
  23. CREATE PROCEDURE proc1(id int,sex_str varchar(8),name varchar(16))
  24. BEGIN
  25. /*声明变量v_sex⽤于存放性别*/
  26. DECLARE v_sex TINYINT UNSIGNED;
  27. /*根据sex_str的值来设置性别*/
  28. CASE sex_str
  29. when '男' THEN
  30. SET v_sex = 1;
  31. WHEN '⼥' THEN
  32. SET v_sex = 2;
  33. END CASE ;
  34. /*插⼊数据*/
  35. INSERT INTO t_user VALUES (id,v_sex,name);
  36. END $
  37. /*结束符置为;*/
  38. DELIMITER;
  39. CALL proc1(6,'男','郭富城');
  40. -- 示例3:函数中使⽤
  41. /*删除存储过程proc1*/
  42. DROP FUNCTION IF EXISTS fun1;
  43. /*声明结束符为$*/
  44. DELIMITER $
  45. /*创建存储过程proc1*/
  46. CREATE FUNCTION fun1(sex TINYINT UNSIGNED)
  47. RETURNS varchar(8)
  48. BEGIN
  49. /*声明变量v_sex⽤于存放性别*/
  50. DECLARE v_sex VARCHAR(8);
  51. CASE sex
  52. WHEN 1 THEN
  53. SET v_sex:='男';
  54. ELSE
  55. SET v_sex:='⼥';
  56. END CASE;
  57. RETURN v_sex;
  58. END $
  59. /*结束符置为;*/
  60. DELIMITER ;
  61. select sex, fun1(sex) 性别,name FROM t_user;
  62. -- CASE结构语法第2种⽤法
  63. case
  64. when 条件1 then 结果1或者语句1(如果是语句需要加分号)
  65. when 条件2 then 结果2或者语句2
  66. ...
  67. else 结果n或者语句n
  68. end [case] (如果是放在begin end之间需要加case,如果是在select后⾯case可以省
  69. 略)
  70. -- if结构语法(只能使⽤在begin end之间)
  71. if 条件语句1 then 语句1;
  72. elseif 条件语句2 then 语句2;
  73. ...
  74. else 语句n;
  75. end if;
  76. -- 示例:写⼀个存储过程,实现⽤户数据的插⼊和新增,如果id存在,则修改,不存在则新增,并返回结果
  77. /*删除id=7的记录*/
  78. DELETE FROM t_user WHERE id=7;
  79. /*删除存储过程*/
  80. DROP PROCEDURE IF EXISTS proc2;
  81. /*声明结束符为$*/
  82. DELIMITER $
  83. /*创建存储过程*/
  84. CREATE PROCEDURE proc2(v_id int,v_sex varchar(8),v_name
  85. varchar(16),OUT result TINYINT)
  86. BEGIN
  87. DECLARE v_count TINYINT DEFAULT 0;/*⽤来保存user记录的数量*/
  88. /*根据v_id查询数据放⼊v_count中*/
  89. select count(id) into v_count from t_user where id = v_id;
  90. /*v_count>0表⽰数据存在,则修改,否则新增*/
  91. if v_count>0 THEN
  92. BEGIN
  93. DECLARE lsex TINYINT;
  94. select if(lsex='男',1,2) into lsex;
  95. update t_user set sex = lsex,name = v_name where id = v_id;
  96. /*获取update影响⾏数*/
  97. select ROW_COUNT() INTO result;
  98. END;
  99. else
  100. BEGIN
  101. DECLARE lsex TINYINT;
  102. select if(lsex='男',1,2) into lsex;
  103. insert into t_user VALUES (v_id,lsex,v_name);
  104. select 0 into result;
  105. END;
  106. END IF;
  107. END $
  108. /*结束符置为;*/
  109. DELIMITER ;
  110. -- while循环语法
  111. -- 标签:是给while取个名字,标签和iterateleave结合⽤于在循环内部对循环
  112. 进⾏控制,如:跳出循环、结束本次循环
  113. -- 这个循环先判断条件,条件成⽴之后,才会执⾏循环体,每次执⾏都会先进⾏判断
  114. [标签:]while 循环条件 do
  115. 循环体
  116. end while [标签];
  117. -- 示例:添加iterate控制语句
  118. /*删除test1表记录*/
  119. DELETE FROM test1;
  120. /*删除存储过程*/
  121. DROP PROCEDURE IF EXISTS proc5;
  122. /*声明结束符为$*/
  123. DELIMITER $
  124. /*创建存储过程*/
  125. CREATE PROCEDURE proc5(v_count int)
  126. BEGIN
  127. DECLARE i int DEFAULT 0;
  128. a:WHILE i<=v_count DO
  129. SET i=i+1;
  130. /*如果i不为偶数,跳过本次循环*/
  131. IF i%2!=0 THEN
  132. ITERATE a;
  133. END IF;
  134. /*插⼊数据*/
  135. INSERT into test1 values (i);
  136. END WHILE;
  137. END $
  138. /*结束符置为;*/
  139. DELIMITER ;
  140. -- repeat循环语法
  141. -- repeat不管如何循环都会先执⾏⼀次,然后再判断结束循环的条件,不满⾜结束条件,循环体继续执⾏。这块和while不同,while是先判断条件是否成⽴再执⾏循环体
  142. [标签:]repeat
  143. 循环体;
  144. until 结束循环的条件 end repeat [标签];
  145. -- 示例:⽆循环控制语句
  146. /*删除存储过程*/
  147. DROP PROCEDURE IF EXISTS proc6;
  148. /*声明结束符为$*/
  149. DELIMITER $
  150. /*创建存储过程*/
  151. CREATE PROCEDURE proc6(v_count int)
  152. BEGIN
  153. DECLARE i int DEFAULT 1;
  154. a:REPEAT
  155. INSERT into test1 values (i);
  156. SET i=i+1;
  157. UNTIL i>v_count END REPEAT;
  158. END $
  159. /*结束符置为;*/
  160. DELIMITER ;
  161. -- loop循环语法
  162. -- 相当于⼀个死循环,需要在循环体中使⽤iterate或者leave来控制循环的执⾏
  163. [标签:]loop
  164. 循环体;
  165. end loop [标签];
  166. -- 示例:⽆循环控制语句
  167. /*删除存储过程*/
  168. DROP PROCEDURE IF EXISTS proc7;
  169. /*声明结束符为$*/
  170. DELIMITER $
  171. /*创建存储过程*/
  172. CREATE PROCEDURE proc7(v_count int)
  173. BEGIN
  174. DECLARE i int DEFAULT 0;
  175. a:LOOP
  176. SET i=i+1;
  177. /*当i>v_count的时候退出循环*/
  178. IF i>v_count THEN
  179. LEAVE a;
  180. END IF;
  181. INSERT into test1 values (i);
  182. END LOOP a;
  183. END $
  184. /*结束符置为;*/
  185. DELIMITER ;
  • 总结
    • if函数常⽤在select中
    • case语句有2种写法,主要⽤在select、begin、end中,select中end后⾯可以省略case,begin end中使⽤不能省略case
    • if语句⽤在begin end中
    • 3种循环体的使⽤,while类似于java中的while循环,repeat类似于java中的do while循环,loop类似于java中的死循环,都⽤于begin end中
    • 循环中体中的控制依靠leave和iterate,leave类似于java中的break可以退出循环,iterate类似于java中的continue可以结束本次循环