MySQL高级 一: 视图&存储过程&函数&触发器
视图
概述
- 视图是基于查询的虚拟表。通俗的理解,视图就是一条 SELECT 语句执行后返回的结果集。
- SELECT 语句所查询的表称为视图的基表,而查询的结果集称为虚拟表,视图本身并不存储具体的数据,视图的数据存在于视图的基表中,基本表数据发生 了改变,视图的数据也会跟着改变。
使用视图的好处
使用视图是为了方便复杂的查询语句。基本思路是将复杂的查询语句定义在 视图内部,然后对视图进行查询,从而简化复杂的查询语句。
视图的语法
-
定义视图
-- 将select 查询到的结果包装在视图中,可以重复使用 CREATE VIEW 视图名 AS select查询语句;
-
删除视图
drop view 视图名
-
使用视图
SELECT * FROM 视图名
示例代码
-- 创建视图(查询"每个管理员对应的角色"的结果作为视图)
CREATE VIEW view_role AS
SELECT r.id,
a.account,
r.name
FROM ROLE r
LEFT JOIN admin_role ON r.id=role_id
LEFT JOIN admin a ON admin_id=a.id
-- 使用视图
SELECT * FROM view_role
-- 删除视图
DROP VIEW view_role
总结:视图就是将一个较为复杂的查询语句的结果包装在视图中,简化语句,视图被存储在数据库中,可以重复使用,视图中并不存储数据
存储过程
概述
如果实现用户的某些需求时,需要编写一组复杂的 SQL 语句才能实现,那么 可以将这组复杂的 SQL 语句集编写在数据库中,由 JDBC 调用来执行这组 SQL 语句。把编写在数据库中的 SQL 语句集称为存储过程。
存储过程(PROCEDURE)是事先经过编译并存储在数据库中的一段 SQL 语 句的集合。调用存储过程可以简化应用开发人员的很多工作,减少数据在数据库和应用服务器之间的传输,对于提高数据处理的效率是有好处的。
存储过程类似于 JAVA 语言中的方法,需要先定义,使用时需要调用。存储过程可以定义参数,参数分为 IN、OUT、INOUT 类型三种类型。
- IN 类型的参数表示接收调用者传入的数据
- OUT 类型的参数表示向调用者返回的数据
- INOUT 类型的参数既可以接收调用者传入的参数,也可以向调用者返回数据。
存储过程定义
- 创建存储过程
DELIMITER $$ -- 开始位置
CREATE
PROCEDURE 存储过程名([in 变量名 类型,out 参数 2,…])
BEGIN
[declare 变量名 类型 [DEFAULT 值];]
存储过程语句块;
END$$
DELIMITER; -- 结束位置
- 调用存储过程
CALL 存储过程名(参数列表);
- 删除存储过程
DROP PROCEDURE [ IF EXISTS ] <过程名>
解析:
- 存储过程的参数分为 in(入参),out(出参),inout(可出可入) 三种类型。
- in 代表输入参数(默认情况下为 in 参数),表示该参数的值必须由调用程序指定。
- out 代表输出参数,表示该参数的值经存储过程计算后,将 out 参数的计算结果返回给调用程序。
- inout 代表即是输入参数,又是输出参数,表示该参数的值即可以由调用程序 指定,又可以将 inout 参数的计算结果返回给调用程序
- 存储过程中的语句必须包含在 begin 和 end 之间。
- declare 中用来声明变量,变量默认赋值使用 default,语句块中改变变量值,使用 set 变量=值;
存储过程的使用
不带参数的存储过程
DELIMITER $$
CREATE
-- 在ssm数据库中创建一个名为demo1的存储过程,类似java中的方法,可以入参出参
PROCEDURE `ssm`.`demo1`()
BEGIN
-- DECLARE 声明变量
DECLARE v_name VARCHAR(20) DEFAULT 'sql';
SET v_name = 'mysql';
-- 测试输出
SELECT v_name;
END$$
DELIMITER ; -- 结束位置
调用存储过程
CALL demo1();
带参数的存储过程
-- 定义一个有参数的存储过程
DELIMITER $$
CREATE
PROCEDURE ssm.demo2(IN p_type INT,OUT p_count INT) -- IN 入参,OUT 出参
BEGIN
-- 根据传入的新闻类型参数-->统计指定新闻类型的数量
SELECT COUNT(*) INTO p_count FROM news WHERE TYPE=p_type;
SELECT p_count; -- 查询出参结果
END$$
DELIMITER ;
-- @p_count 用来测试输出参数用,实际开发时不是这样
CALL demo2(2,@p_count);
流程控制语句 if else
语法:
IF 表达式1 THEN
表达式1成立时执行的语句;
ELSEIF 表达式2 THEN
表达式2成立时执行的语句;
ELSE
都不成立时执行的语句;
END IF; -- 结束if语句
在存储过程中使用IF
-- 测试if语句的存储过程
DELIMITER $$
CREATE
PROCEDURE `ssm`.`test_if`(IN p_day INT,OUT p_name VARCHAR(10)) -- IN 入参,OUT 出参
BEGIN
IF p_day=1 THEN
SET p_name='星期一';
SELECT p_name;
ELSEIF p_day=2 THEN
SET p_name='星期二';
SELECT p_name;
ELSE
SET p_name='其他日期';
SELECT p_name;
END IF;
END$$
DELIMITER ;
-- 调用
CALL test_if(1,@p_day);
mybaits调用存储过程
mysql代码
-- 存储过程-->报存管理员,在存储过程中判断是否存在此账号,存在返回该账号,不存在就存到表中
DELIMITER $$
CREATE
PROCEDURE `mybatis_db`.`is_exist`(IN p_account VARCHAR(12),IN p_password VARCHAR(12),OUT p_result VARCHAR(12)) -- IN 入参,OUT 出参
BEGIN
-- 声明一个变量 , 接收查询到的结果
DECLARE v_count INT DEFAULT 0;
SELECT COUNT(*) INTO v_count FROM admin WHERE account = p_account;
IF v_count = 0 THEN
INSERT INTO admin(account,PASSWORD)VALUES(p_account,p_password);
SET p_result = '保存成功!';
ELSE
SET p_result = '账号已存在!';
SELECT p_result;
END IF;
END$$
DELIMITER ;
-- 调用
CALL is_exist('1111','111',@p_result);
java代码
@Test
public void save(){
SqlSession sqlSession = MybatisUtil.getSqlSession();
AdminMapper mapper = sqlSession.getMapper(AdminMapper.class);
Map<String,String> map = new HashMap<>();
map.put("account", "a01");
map.put("password", "111");
mapper.save(map);
System.out.println(map.get("result"));//获取输出参数
sqlSession.commit();
sqlSession.close();
}
sql映射文件(AdminMapper.xml)–>必须指定jdbcType
<!-- 必须指定jdbcType
property:java参数 jdbcType:数据库数据类型 mode:IN/OUT 表示入参/出参-->
<parameterMap id="saveMap" type="hashmap">
<parameter property="account" jdbcType="VARCHAR" mode="IN"></parameter>
<parameter property="password" jdbcType="VARCHAR" mode="IN"></parameter>
<parameter property="result" jdbcType="VARCHAR" mode="OUT"></parameter>
</parameterMap>
<!-- 测试存储过程 -->
<insert id="save" parameterMap="saveMap" statementType="CALLABLE">
<!-- call 存储过程名() 调用存储过程 -->
call is_exist(?,?,?);
</insert>
函数
函数类似于存储过程,但是函数主要用于查询
函数使用语法
- 创建
DELIMITER $$
CREATE FUNCTION 函数名([参数列表]) RETURNS 数据类型
BEGIN
DECLARE 变量;
sql 语句;
RETURN值;
END $$
DELIMITER ;
- 删除
DROP FUNCTION 函数名;
使用注意事项
- 参数列表包含两部分:参数名 参数类型
- 函数体:肯定会有 return 语句,如果没有会报错
- 函数体中仅有一句话,则可以省略 begin end
- 使用 delimter 语句设置结束标记
设置函数可以没有参数
-- 全局设置--设置该函数可以没有参数
SET GLOBAL log_bin_trust_function_creators=TRUE
函数使用实例
无参函数
-- 函数
DELIMITER $$
CREATE FUNCTION ssm.fun1() RETURNS INT
BEGIN
-- 声明一个int变量, 变量以v_开始,v表示variable
DECLARE v_count INT DEFAULT 0;
-- 统计admin表中数据量总数
SELECT COUNT(*) INTO v_count FROM admin;
-- 返回总数
RETURN v_count;
END $$
DELIMITER ;
-- 调用
SELECT fun1();
有参函数
(1)写一个区分超管与管理员的函数
-- 写一个区分超管与管理员的函数
DELIMITER $$
CREATE FUNCTION ssm.find_type(p_type INT) RETURNS VARCHAR(10)
BEGIN
DECLARE v_type VARCHAR(10) DEFAULT '';
IF p_type = 0 THEN
SET v_type = '超级管理员';
ELSE
SET v_type = '管理员';
END IF;
RETURN v_type;
END$$
DELIMITER ;
-- 调用 (查询amdin时,将admin的类型id对应的类型名称查询到)
SELECT account,id,find_type(TYPE) FROM admin
-- 创建一个查询新闻类型的函数
DELIMITER $$
CREATE FUNCTION find_news_type(p_typeid INT) RETURNS VARCHAR(10)
BEGIN
DECLARE v_type VARCHAR(10) DEFAULT '';
SELECT NAME INTO v_type FROM news_type WHERE id = p_typeid;
RETURN v_type;
END$$
DELIMITER ;
-- 调用
SELECT id,title,find_news_type(TYPE) FROM news;
触发器(trigger)
概述
触发器(trigger)是一种特殊的存储过程,其特殊性在于它并不需要用户直接调用,而是在对表操作(添加、修改、删除)之前或者之后自动执行的存储过程。
类似于Java/JavaScript中的事件,在指定条件触发时执行
触发器的特点
-
与表相关联
触发器定义在特定的表上,这个表称为触发器表。 -
自动激活触发器
当对表中的数据执行 INSERT、UPDATE 或 DELETE 操作时,如果对表上的这个特定操作定义了触发器,该触发器自动执行,这是不可撤销的。 -
不能直接调用
与存储过程不同,触发器不能被直接调用,也不能传递或接受参数。 -
作为事务的一部分
触发器与激活触发器的语句一起做为对一个单一的事务来对待,可以从触发器中的任何位置回滚。
触发器语法
-- 触发器
DELIMITER $$
CREATE TRIGGER 触发器名称 触发时机 触发事件
ON 表名称
FOR EACH ROW -- 行级触发器 操作多行时,每行都会触发 '触发器'
BEGIN
执行的sql语句;
END$$
DELIMITER ;
语法解析
- 触发器名称:是用来标识触发器的,由用户自定义。
- 触发时机:其值是 before 或 after。
- 触发事件:其值是 insert,update 和 delete
- 表名称:标识建立触发器的表名,即在哪张表上建立触发器
- 语句:是触发器程序体,触发器程序可以使用 begin 和 end 作为开始和结束, 中间包含多条语句;
案例1–删除admin时,自动触发删除admin与role的关系
-- 触发器
DELIMITER $$
CREATE TRIGGER ssm.delete_admin_role
BEFORE DELETE
ON admin
FOR EACH ROW -- 行级触发器 操作多行时,每行都会触发 '触发器'
BEGIN
DELETE FROM admin_role WHERE admin_id = old.id; -- old 旧的数据
END$$
DELIMITER ;
-- 删除之前会触发触发器删除admin与role表的关联关系
DELETE FROM admin WHERE id = 39
案例2–新增admin时,自动向admin_log表插入数据
-- 向admin表插入数据时,往admin_log中插入记录
DELIMITER $$
CREATE TRIGGER save_admin_log
AFTER INSERT ON admin
FOR EACH ROW -- 行级触发器 操作多行时,每行都会触发 '触发器'
BEGIN
INSERT INTO admin_log(id,account,oper_time)VALUES(new.id,new.account,NOW());
END$$
DELIMITER ;
INSERT INTO admin(account,PASSWORD)VALUES('b01','111')
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/15604.html