【MySQL】游标和触发器

一、游标

1.1 什么是游标

1、使用背景
在我们使用update或者delete操作数据时,一般都会根据条件语句查询出很多条记录组成的数据集,然后一次性批量操作

假设我们想要对这个结果集中的数据 一行一行的进行操作,比如某个条件满足后,就不继续往下操作了,这个时候就要用到游标了

游标可以在存储过程和函数中使用

2、使用步骤

  • 声明游标(位置放在变量声明语句的后面,因为变量声明必须紧跟begin)
    MySQL、DB2、SQL server和Maria DB中的声明语法为
    DECLARE cursor_name CURSOR FOR 查询语句;
    Oracle、Postgresql中的声明语法为
    DECLARE cursor_name CURSOR IS 查询语句;
  • 打开游标
    OPEN cursor_name;
  • 使用游标:从游标中获取数据(注意:var_name必须在声明游标之前就定义好)
    FETCH cursor_name INTO var_name,var_name2...;
    这句话的作用是使用cursor_name来读取当前行,并将数据保存到变量var_name中,游标指针指向下一行,如果读取的数据行有多个字段,那就在INTO关键字后赋值给多个变量
  • 关闭游标
    CLOSE cursor_name;

3、使用案例
新建一个博客表t_blog_view,设定一个值num
看看最少需要累加多少篇博客的浏览量才能达到这个值(根据浏览量做一个降序排列)

CREATE TABLE `t_blog_view`  (`blog_name` varchar(50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '博客名称',`blog_author` varchar(50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '博客作者',`blog_views` int(20) NOT NULL COMMENT '博客浏览量'
) ENGINE = InnoDB CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;INSERT INTO `t_blog_view` VALUES ( 'Mybatis系列', 'Decade0712', 2000);
INSERT INTO `t_blog_view` VALUES ( 'Spring系列', 'Decade0712', 4000);
INSERT INTO `t_blog_view` VALUES ( '设计模式系列', 'Decade0712', 6000);
INSERT INTO `t_blog_view` VALUES ( 'JVM系列', '十年', 8000);
INSERT INTO `t_blog_view` VALUES ( 'MySQL基础', 'Decade0712', 4000);
INSERT INTO `t_blog_view` VALUES ( 'Java8新特性', '十年', 5500);

我们定义一个存储过程进行测试

DELIMITER //CREATE PROCEDURE test_cursor(IN num DOUBLE,OUT res_count INT)
BEGIN# 声明局部变量DECLARE sum_views DOUBLE DEFAULT 0.0; # 记录累加浏览量DECLARE current_views DOUBLE;  # 记录当前博客浏览量DECLARE blog_count INT DEFAULT 0; # 记录累加博客数# 1、声明游标DECLARE blog_cursor CURSOR FOR SELECT blog_views from t_blog_view ORDER BY blog_views DESC;# 2、打开游标OPEN blog_cursor;# 3、使用游标,因为要累加所以使用循环语句REPEATFETCH blog_cursor INTO current_views;SET sum_views = sum_views + current_views;SET blog_count = blog_count + 1;UNTIL sum_views >=  numEND REPEAT;# 把累加的博客数赋值给输出变量SET res_count = blog_count;# 4、关闭游标CLOSE blog_cursor;
END //DELIMITER ;# 进行调用
CALL test_cursor(10086,@res_count);
SELECT  @res_count;# 删除存储过程
DROP PROCEDURE test_cursor;

结果如下
在这里插入图片描述
4、优缺点

  • 优点:使用游标,能够逐条读取结果集中的数据
  • 缺点:使用游标,会对数据进行加锁,在业务并发量大时,会影响业务的效率,并且会消耗系统内存资源

二、触发器

1、使用场景
假设我们现在有2张表商品表和库存表,我们新引入一种商品时,除了要修改商品表的数据,还要修改库存表的数据

为了保证不遗漏任何一个动作,我们一般使用事务将其包裹起来,使这两个动作成为一个原子操作

或者使用触发器,让商品表数据插入的动作自动触发库存表数据插入的动作

2、概述
触发器是由事件来触发某个操作,包括INSERT、UPDATE、DELETE事件。事件就是指用户的动作或者触发某项行为。如果定义了触发程序,当数据库执行这些语句的时候就相当于事件发生了,就会自动激发触发器去执行相应的操作

3、触发器的创建

CREATE TRIGGER 触发器名
{BEFORE|AFTER} {INSERT|UPDATE|DELETE} ON 表名
FOR EACH ROW
触发器执行的语句块;
  • 表名:触发器监控的对象
  • BEFORE|AFTER:表示触发的时间,是在事件发生之前还是发生之后
  • INSERT|UPDATE|DELETE:表示触发的事件,是插入数据事件、更新数据事件还是删除数据事件
  • FOR EACH ROW:每操作一条表中的数据,就激发一次触发器
  • 触发器执行的语句块:可以是单条语句块,也可以是BEGIN...END包裹的复杂语句块

4、举例
我们新建2张表,test_triggle和test_triggle_log,每当我们要向test_triggle中插入数据时,先在test_triggle_log中记录日志

CREATE TABLE test_triggle(
id INT PRIMARY KEY AUTO_INCREMENT,
t_note VARCHAR(30)
);CREATE TABLE test_triggle_log(
id INT PRIMARY KEY AUTO_INCREMENT,
t_log VARCHAR(30)
);

创建触发器

DELIMITER //
CREATE TRIGGER test_trigger_tir
BEFORE INSERT ON test_triggle
FOR EACH ROW
BEGININSERT INTO test_triggle_log(t_log)VALUES('before insert to...');
END //
DELIMITER ;INSERT INTO test_triggle(t_note)
VALUES('test insert DEMO1...');SELECT * FROM test_triggle;
SELECT * FROM test_triggle_log;

结果如下,test_triggle_log表中自动增加了一条记录
在这里插入图片描述
案例二:我们向表test_triggle中插入数据前,需要先对这条记录的id做一个判断,如果是双数,就报错,否则就正常插入

注意:我们使用NEW表示INSERT要插入的那条数据,要获取某个字段,使用NEW.column即可
使用OLD表示DELETE要删除的那条数据

DELIMITER //
CREATE TRIGGER test_trigger_tir_single_num
BEFORE INSERT ON test_triggle
FOR EACH ROW
BEGIN# 创建一个变量,将当前记录的id对2取余的结果赋值给它,以便于后续做单双数的判断DECLARE is_single INT;SELECT NEW.id % 2 INTO is_single;IF is_single = 0 THENSIGNAL SQLSTATE 'HY000' SET MESSAGE_TEXT = 'id为双数无法插入';END IF;
END //
DELIMITER ;# 测试
INSERT INTO test_triggle(id,t_note) VALUES(1,'test insert DEMO1...');
INSERT INTO test_triggle(id,t_note) VALUES(2,'test insert DEMO2...');# 查看test_triggle表是否正常插入数据
SELECT * from test_triggle;

结果如下,id为双数时,无法插入数据
在这里插入图片描述
在这里插入图片描述

5、查看触发器

# 查看所有触发器的定义
SHOW TRIGGERS;# 查看指定触发器的定义
SHOW CREATE TRIGGER 触发器名称;# 从系统库information_schema中查看触发器的定义
SELECT * FROM information_schema.TRIGGERS;

6、删除触发器

DROP TRIGGER IF EXISTS 触发器名称;

7、触发器的优缺点

  • 优点:
    • 可以保证数据的完整性
    • 可以通过触发器帮助我们记录数据的操作日志
    • 可以在操作数据前,对数据的合法性做一个检验
  • 缺点:
    • 可读性差
    • 相关数据的变更,可能会导致触发器报错

8、注意点
如果在子表中定义了外键约束,且子表的中又定义了基于此表UPDATE或者DELETE操作的触发器
如果父表数据修改引发了子表的数据变化,在这种情况下并不会激发触发器

如有错误,欢迎指正!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/807682.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

6 Reverse Linked List

分数 20 作者 陈越 单位 浙江大学 Write a nonrecursive procedure to reverse a singly linked list in O(N) time using constant extra space. Format of functions: List Reverse( List L ); where List is defined as the following: typedef struct Node *PtrToNo…

IJKPLAYER源码分析-OpenSL ES播放

1 前言 与IJKPLAYER处理AudioTrack播放类似,OpenSL ES的接入需要满足SDL_Aout的接口规范,所不同的是OpenSL ES播放是在native完成的,调用的是NDK接口OpenSL ES的播放能力。关于OpenSL ES的详细介绍,请参考官方文档 OpenSL ES 一文…

《策略模式(极简c++)》

本文章属于专栏- 概述 - 《设计模式(极简c版)》-CSDN博客 本章简要说明适配器模式。本文分为模式说明、本质思想、实践建议、代码示例四个部分。 模式说明 方案:策略模式是一种行为设计模式,它定义了一系列算法,将每…

分布式任务调度平台 XXL-JOB

官网链接:https://www.xuxueli.com/xxl-job/ 容器发布命令: docker run -e \PARAMS"--xxl.job.accessToken??? \--server.servlet.context-path/??? \--spring.datasource.username??? \--spring.datasource.password??? \--sprin…

SOCKS5代理、代理IP、HTTP与网络安全的综合视角

在当今互联网时代,网络安全已成为信息技术领域中的一个重要议题。作为一名软件工程师,深入理解SOCKS5代理、代理IP、HTTP协议以及网络安全的细微差别和内在联系,对于设计和维护安全的网络系统至关重要。本文将从技术深度出发,探讨…

Lora 串口透传开发 5

1 简介 串口转usb、转wifi等很多应用 2 设计原理 2.1 设计需求 1将LoRa终端定义成两种角色:Master和Slave 2一个模块发送任意字节长度(小于128Byte)数据,另一模块都可以接收到 3PC机上通过串口调试助手实现接收和发送 4终端在LCD屏幕上显…

智慧公厕升级为多功能城市智慧驿站,助力智慧城市发展

在现代城市的建设中,公共厕所作为基础必备的民生设施,一直是城市管理的重要组成部分。随着科技的不断发展,智慧公厕应运而生,成为了公共厕所信息化、数字化、智慧化的应用解决方案。而近年来,智慧公厕也进行了升级发展…

损失函数:BCE Loss(二元交叉熵损失函数)、Dice Loss(Dice相似系数损失函数)

损失函数:BCE Loss(二元交叉熵损失函数)、Dice Loss(Dice相似系数损失函数) 前言相关介绍BCE Loss(二元交叉熵损失函数)代码实例直接计算函数计算 Dice Loss(Dice相似系数损失函数&a…

分布式 SpringCloudAlibaba、Feign与RabbitMQ实现MySQL到ES数据同步

文章目录 ⛄引言一、思路分析⛅实现方式⚡框架选择 二、实现数据同步⌚需求分析⏰搭建环境⚡核心源码 三、测试四、源码获取⛵小结 ⛄引言 本文参考黑马 分布式Elastic search Elasticsearch是一款非常强大的开源搜索引擎,具备非常多强大功能,可以帮助…

如何使用Android手机通过JuiceSSH远程访问本地Linux服务器

文章目录 1. Linux安装cpolar2. 创建公网SSH连接地址3. JuiceSSH公网远程连接4. 固定连接SSH公网地址5. SSH固定地址连接测试 处于内网的虚拟机如何被外网访问呢?如何手机就能访问虚拟机呢? cpolarJuiceSSH 实现手机端远程连接Linux虚拟机(内网穿透,手机端连接Linux虚拟机) …

对React Router的理解

对React Router的理解 1. 基本概念2. 核心组件2.1 使用React Router的基本示例:3. 路由守卫和动态路由3.1 路由守卫3.2 动态路由 React Router是React应用中实现路由功能的一个库,它允许开发者在单页应用中创建多页面应用的感觉,通过URL的变化…

Python上解决TypeError: not all arguments converted during string formatting错误

目录 背景尝试1: pymysql模块的escape_string方法尝试2: 修改pandas.read_excel引擎尝试3: 回退xlrd版本总结 背景 在Linux上部署的时候, 使用pandas模块读取Excel, 然后pymysql模块入库, 结果发生了错误 Traceback (most recent call last):File "/usr/local/lib64/pyth…

3月谷歌应用上架/下架情况,上架难度加大,开发者面临新挑战?

在3月份,Google play应用商店应用上架和下架出现了前所未有的情况。很多开发者表示,上架难度简直是“地狱”级别。 下图是3月份美国、巴西、印度、中国香港的下架数量的折线图,根据市场数据监测,可以清晰地看到3月份中旬之后&…

DESTINATION MOON 香港站回顾|聆听 Web3 创新者的未来对话

创新者汇聚 Web3 行业,如何才能在生态、技术、投资的发展新风口把握机遇?「TinTin Destination Moon」香港站活动于 4 月 6 日下午如期举行!Web3AI 的融合发展之道在哪?ETF 时代的投资逻辑有哪些?区块链未来的关键究竟…

使用HTML+CSS实现一个简单的登录页面

整个项目使用文件&#xff1a; HTML代码部分&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><ti…

【SQL】数据定义语言(DDL):包括创建、修改和删除数据库对象

数据定义语言&#xff08;DDL&#xff09;是用于定义和管理数据库中的数据结构和对象的SQL语言子集。它允许用户创建、修改和删除数据库中的表、索引、触发器、序列、存储过程等对象。DDL语句在数据库系统中执行时&#xff0c;通常会影响整个数据库的结构&#xff0c;而不是单个…

如何理解单片机 pwm 控制的基本原理?

单片机PWM&#xff08;脉宽调制&#xff09;控制的基本原理&#xff0c;简而言之&#xff0c;就是通过改变脉冲信号的宽度&#xff08;占空比&#xff09;来控制模拟电路。这涉及到单片机生成一系列脉冲信号&#xff0c;每个脉冲信号的高电平持续时间和整个周期的比值&#xff…

计算机视觉 | 基于二值图像数字矩阵的距离变换算法

Hi&#xff0c;大家好&#xff0c;我是半亩花海。本实验基于 OpenCV 实现了二值图像数字矩阵的距离变换算法。首先生成一个 480x480 的黑色背景图像&#xff08;定义黑色为0&#xff0c;白色为1&#xff09;&#xff0c;在其中随机选择了三个白色像素点作为距离变换的原点&…

MicroPython with LVGL

官方博客:Micropython LittlevGL | LVGL’s Blog github:GitHub - lvgl/lv_micropython: Micropython bindings to LVGL for Embedded devices, Unix and JavaScript 官方在线模拟器:https://sim.lvgl.io/(需要电脑能访问外网才能使用) 电脑不能访问外网会出现以下错误&…

JVM内存区域

类加载 将class文件加载到方法区中 验证&#xff1a;验证待加载的class文件是否正确&#xff0c;比如验证文件的格式 准备&#xff1a;为static变量分配内存并赋零值 解析&#xff1a;将符号引用解析为直接引用 类加载器 双亲委派 总结就是&#xff0c;向上查找有没有加载过…