MySQL双层游标嵌套循环方法

文章目录

    • 1、需求描述
    • 2、思路
    • 3、创建存储过程


1、需求描述

1、在项目中,需要将A表中主键id,逐个取出,作为条件,在B表中去逐一查询,将B表查询到的结果集(A表B表关系:一对多),逐一遍历,连同A表的id拼接运算,逐个插入到C表中。

2、 在Java中很容易实现,A表获取到的结果集,循环遍历取出id,去B表查询;遍历B表结果集,插入到C表中。 相当于2个循环,即可实现需求。 这样会有一个问题,频繁连接数据库,造成大量资源开销。 那么在存储过程中,该怎么实现呢?

2、思路

  要实现逐行获取数据,需要用到MySQL中的游标,一个游标相当于一个for循环,这里需要用到2个游标。如何在MySQL中实现游标双层循环呢?

3、创建存储过程

CREATE DEFINER=`root`@`%` PROCEDURE `student`()
BEGIN-- 定义变量-- 假设有一张学生表,有id,student_name字段DECLARE outer_done INT DEFAULT FALSE; -- 外层游标控制变量DECLARE studentTableId int;    -- 学生表IDdeclare studentTableName VARCHAR(100);   -- 学生姓名declare outer_cursor cursor for select id,student_name from student_table  where `disable` = '0'; DECLARE CONTINUE HANDLER FOR NOT FOUND SET outer_done = TRUE;open outer_cursor;     while not outer_done dofetch outer_cursor into studentTableId,studentTableName;  -- 这里循环取值,赋值到上面定义的变量中-- 开始定义内层游标BEGIN -- inner BEGIN-- 假设有一张成绩表,包含字段id,student_name,score字段DECLARE scoreTableId int;    -- 成绩Iddeclare scoreTableName VARCHAR(100);    -- 学生名字declare scoreTableScore float;   -- 学生分数DECLARE inner_done int DEFAULT FALSE ;DECLARE my_value VARCHAR(255);declare inner_cursor cursor for select id,student_name,score from score_table where `disable` = '0'; DECLARE CONTINUE HANDLER FOR NOT FOUND SET inner_done  = TRUE ;OPEN inner_cursor; -- 打开内层游标WHILE not inner_done DO -- inner WHILEFETCH inner_cursor INTO scoreTableId,scoreTableName,scoreTableScore ; -- 从【内层游标】中获取数据,赋值到定义变量中IF studentTableName = scoreTableName THEN    -- 判断名字一样(测试,生产不要用名称进行判断)set my_value = CONCAT_WS('-',studentTableName,scoreTableScore);    -- 给变量赋值 CONCAT_WS函数可以按照固定的连接符,将数据进行连接,例如 张三-95select my_value;     -- 打印变量值END IF;-- 假设有一张汇总表(summary_table),将处理的数据进行更新update summary_table set summary_column=my_value where summary_table_student_id=studentTableId;END WHILE ; -- END inner WHILECLOSE inner_cursor; -- 循环结束后,关闭内层游标END; -- END inner BEGINend while;        close outer_cursor;  
END

看图清晰一点。

在这里插入图片描述
  到这里就完成了,存储过程里面的注释很详细,就不多赘述了,我在写存储过程中也是踩了不少坑,记录下来,希望能帮到各位coder。

  有问题欢迎评论区留言或者私信,我看到就会回复。

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

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

相关文章

upload-labs通关

前记: 在这里面我们使用一句话木马时使用php里的一个函数phpinfo(),该函数能显示出网页具体的php版本和有关的信息。 pass-01(js前端验证) 方法1:禁用js/删除js验证 1.禁用js 按f12&#xff…

MySQL/MariaDB 如何查看当前的用户

MySQL 的所有数据库用户信息是存储在 user 数据表中的。 可以在登录成功数据后运行 SQL: MariaDB [(none)]> select user,host from user;就可以查看到数据中的所有用户信息。 MariaDB [(none)]> select user,host from user; ERROR 1046 (3D000): No databa…

笔记:编写程序,分别采用面向对象和 pyplot 快捷函数的方式绘制正弦曲线 和余弦曲线。 提示:使用 sin()或 cos()函数生成正弦值或余弦值。

文章目录 前言一、面向对象和 pyplot 快捷函数的方式是什么?二、编写代码面向对象的方法:使用 pyplot 快捷函数的方法: 总结 前言 本文将探讨如何使用编程语言编写程序,通过两种不同的方法绘制正弦曲线和余弦曲线。我们将分别采用…

美国洛杉矶站群服务器如何提高网站排名?

美国洛杉矶站群服务器怎么样?美国洛杉矶站群服务器如何提高网站排名?Rak部落小编为您整理发布美国洛杉矶站群服务器如何提高网站排名? 美国洛杉矶站群服务器可以通过以下几种方式帮助提高网站排名: - **提升网站性能**:美国站群服务器通常配备高速CPU…

cocos-lua资源管理

本文介绍cocos-lua项目的资源管理和工作流,适用人群包括初学者和有经验开发者,故读者可根据自己的需要有选择性的查阅自己需要的内容 一.简单案例解析 下文通过介绍一个简单demo,介绍合图和资源目录结构 1.1 运行效果 1.2 ccs结构 1.3 目录…

Kotlin基础​​

数据类型 定义变量 var表示定义变量,可以自动推导变量类型,所以Int可以不用写。 定义常量 条件语句 if表达式可以返回值,该值一般写在if里的最后一行 类似switch的用法 区间 循环 a是标签,可以直接break到标签的位置&#xf…

【HarmonyOS】Stage 模型 - 基本概念

一、项目结构 如图1所示: 图1 从项目结构来看,这个应用的内部包含了一个子模块叫 entry,模块是应用的基本功能单元,它里面包含源代码、资源、配置文件等。 像这样的模块在应用内部可以创建很多。但模块整体来讲就分成两大类&am…

Docker基本操作 Linux里边操作

docker镜像操作命令: docker images:查看所有镜像; docker rmi:删除镜像 后边可以跟镜像的名字或者id指定要删除的镜像; docker pull:拉取镜像; docker push:推送镜像到服务; docker save :打包镜像 后边有用法; docker load:加载镜像&…

Centos/linux根目录扩容、分区、挂载。LVM、物理卷、逻辑卷

前言    (空格) :分区挂载和扩容是两码事 每个Linux使用者在安装Linux时都会遇到这样的困境:在为系统分区时,如何精确评估和分配各个硬盘分区的容量,因为系统管理员不但要考虑到当前某个分区需要的容量&a…

分布式版本控制系统——Git

分布式版本控制系统——Git 一、Git安装二、创建版本库三、将文件交给Git管理四、Git的工作区和暂存区1.工作区(Working Directory)2.版本库 五、版本回退和撤销修改1.版本回退2.撤销修改 六、删除文件七、常用基础命令总结八、参考 分布式版本控制系统&…

javaScript基础2

javaScript 一.运算符二.流程控制1.顺序流程控制2.分支流程控制(1)if/if..else/if多分支(2).三元表达式(4).switch和if else区别 3.循环流程控制(1).for循环/双重for循环(2).一些例子(3).while循环/do..whi…

python使用opencv对图像的基本操作(2)

13.对多个像素点进行操作,使用数组切片方式访问 img[i,:] img[j,:] #将第j行的数值赋值给第i行 img[-2,:]或img[-2] #倒数第二行 img[:,-1] #最后一列 img[50:100,50:100] #50-100行,50-100列(不包括第100行和第100列) img[:100…