存储、游标的简介及嵌套使用
1.存储过程简介
一个存储过程是一个可编程的函数,它在数据库中创建并保存。它可以有SQL语句和一些特殊的控制结构组成。当希望在不同的应用程序或平台上执行相同的函数,或者封装特定功能时,存储过程是非常有用的。数据库中的存储过程可以看做是对编程中面向对象方法的模拟。它允许控制数据的访问方式。
存储过程通常有以下优点:
(1).存储过程增强了SQL语言的功能和灵活性。存储过程可以用流控制语句编写,有很强的灵活性,可以完成复杂的判断和较复杂的运算。
(2).存储过程允许标准组件是编程。存储过程被创建后,可以在程序中被多次调用,而不必重新编写该存储过程的SQL语句。而且数据库专业人员可以随时对存储过程进行修改,对应用程序源代码毫无影响。
(3).存储过程能实现较快的执行速度。如果某一操作包含大量的Transaction-SQL代码或分别被多次执行,那么存储过程要比批处理的执行速度快很多。因为存储过程是预编译的。在首次运行一个存储过程时查询,优化器对其进行分析优化,并且给出最终被存储在系统表中的执行计划。而批处理的Transaction-SQL语句在每次运行时都要进行编译和优化,速度相对要慢一些。
(4).存储过程能过减少网络流量。针对同一个数据库对象的操作(如查询、修改),如果这一操作所涉及的Transaction-SQL语句被组织程存储过程,那么当在客户计算机上调用该存储过程时,网络中传送的只是该调用语句,从而大大增加了网络流量并降低了网络负载。
(5).存储过程可被作为一种安全机制来充分利用。系统管理员通过执行某一存储过程的权限进行限制,能够实现对相应的数据的访问权限的限制,避免了非授权用户对数据的访问,保证了数据的安全。
2.存储过程简单实例
1 CREATE PROCEDURE `procedure_select_pay_success`() -- 创建存储过程 2 BEGIN -- 开始存储过程 3 SELECT * FROM ibsp_balance_account_data WHERE PAY_STATUS="2"; -- 调用的sql内容 4 END -- 结束存储过程 5 6 -- 调用存储过程 7 CALL procedure_select_pay_success(); 8 9 -- 删除存储过程 10 DROP PROCEDURE procedure_select_pay_success;
3.游标简介
游标实际上是一种能从包括多条数据记录的结果集中每次提取一条记录的机制。游标可以被看作是一个查询结果集(可以是零条、一条或由相关的选择语句检索出的多条记录)和结果集中指向特定记录的游标位置组成的一个临时文件,提供了在查询结果集中向前或向后浏览数据、处理结果集中数据的能力。有了游标,用户就可以访问结果集中任意一行数据,在将游标放置到某行之后,可以在该行或从该位置的行块上执行操作。
4.存储过程与游标的使用
1 CREATE PROCEDURE `procedure_delRefunded`() -- 创建存储过程 2 COMMENT '删除ccb_pay_copy中退费记录' -- 给存储过程添加注释 3 begin -- 开始存储过程 4 declare params varchar(100); -- 定义订单号变量 5 DECLARE done INT DEFAULT FALSE; -- 自定义控制游标循环变量,默认false 6 7 DECLARE My_Cursor CURSOR FOR ( SELECT DDH FROM ccb_pay_copy ); -- 定义游标并输入结果集 8 DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE; -- 绑定控制变量到游标,游标循环结束自动转true 9 10 OPEN My_Cursor; -- 打开游标 11 myLoop: LOOP -- 开始循环体,myLoop为自定义循环名,结束循环时用到 12 FETCH My_Cursor into params; -- 将游标当前读取行的数据顺序赋予自定义订单号变量 13 IF done THEN -- 判断是否继续循环 14 LEAVE myLoop; -- 结束循环 15 END IF; 16 -- 自己要做的事情,在 sql 中直接使用自定义变量即可 17 IF params = (select DISTINCT(DDH) from ccb_refund where DDH=params) THEN 18 DELETE FROM ccb_pay_copy WHERE DDH=params; 19 END IF; 20 COMMIT; -- 提交事务 21 END LOOP myLoop; -- 结束自定义循环体 22 CLOSE My_Cursor; -- 关闭游标 23 END -- 结束存储过程
5.嵌套游标的使用(相对于多重循环)
1 CREATE PROCEDURE `procedure_Signle_Pay_Success`() -- 创建存储过程 2 COMMENT '单笔支付成功插入对账表' -- 添加注释 3 begin -- 开始存储过程 4 declare paramsNO varchar(100); -- 定义订单号变量 5 declare paramsTransTime datetime; -- 定义财政执收机变量 6 declare paramsStatus varchar(100); -- 定义决定书编码变量 7 DECLARE done INT DEFAULT FALSE; -- 自定义控制游标循环变量,默认false 8 9 DECLARE My_Cursor CURSOR FOR (SELECT DDH,JYSJ,DDZT from ccb_pay_copy); -- 定义游标并输入结果集 10 DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE; -- 绑定控制变量到游标,游标循环结束自动转true 11 12 OPEN My_Cursor; -- 打开游标 13 myLoop: LOOP -- 开始循环体,myLoop为自定义循环名,结束循环时用到 14 FETCH My_Cursor into paramsNO,paramsTransTime,paramsStatus; -- 将游标当前读取行的数据顺序赋予自定义变量12 15 IF done THEN -- 判断是否继续循环 16 LEAVE myLoop; -- 结束循环 17 END IF; 18 -- 自己要做的事情,在 sql 中直接使用自定义变量即可 19 BEGIN 20 -- IF 0 < (SELECT COUNT(*) FROM ibsp_record_fee where ORDER_NO=paramsNO) THEN 21 declare paramsCardNum varchar(64); -- 定义Fee_Id 22 declare paramsOrg varchar(255); -- 定义财政执收机变量 23 declare paramsDecision varchar(255); -- 定义决定书编码变量 24 declare paramsAmount decimal(64,0); -- 定义支付金额 25 DECLARE done_Fee INT DEFAULT FALSE; -- 自定义控制游标循环变量,默认false 26 DECLARE My_Cursor_Fee CURSOR FOR (SELECT rf.SUM,rf.ORG_UNICODE,ni.CARD_NUM,ni.DECISION_CODE FROM ibsp_record_fee rf LEFT JOIN ibsp_notel_illegal ni on ni.fee_id=rf.id where rf.ORDER_NO=paramsNO); -- 定义游标并输入结果集 27 DECLARE CONTINUE HANDLER FOR NOT FOUND SET done_Fee = TRUE; -- 绑定控制变量到游标,游标循环结束自动转true 28 OPEN My_Cursor_Fee; -- 打开游标 29 myLoopFee: LOOP -- 开始循环体,myLoop为自定义循环名,结束循环时用到 30 FETCH My_Cursor_Fee into paramsAmount,paramsOrg,paramsCardNum,paramsDecision; -- 将游标当前读取行的数据顺序赋予自定义变量12 31 IF done_Fee THEN -- 判断是否继续循环 32 LEAVE myLoopFee; -- 结束循环 33 END IF; 34 INSERT ibsp_balance_account_data (DECISION_CODE,ORG_UNICODE,CARD_NUM,TRANS_DATE,ORDER_NO,ORDER_AMOUNT,TRANS_AMOUNT,TRANS_TIME,PAY_STATUS) VALUES 35 (paramsDecision,paramsOrg,paramsCardNum,paramsTransTime,paramsNO,paramsAmount,paramsAmount,paramsTransTime,paramsStatus); 36 COMMIT; -- 提交事务 37 END LOOP myLoopFee; -- 结束自定义循环体 38 CLOSE My_Cursor_Fee; -- 关闭游标 39 -- END IF; 40 END; 41 COMMIT; -- 提交事务 42 END LOOP myLoop; -- 结束自定义循环体 43 CLOSE My_Cursor; -- 关闭游标 44 END -- 结束存储过程
注意:使用过程中对于查询的时间较长的sql一定要加索引,不然会慢到哭。。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· AI 智能体引爆开源社区「GitHub 热点速览」
· 三行代码完成国际化适配,妙~啊~
· .NET Core 中如何实现缓存的预热?