Navicat中怎么调试存储过程

2024-03-14

这篇文章主要讲解了“Navicat中怎么调试存储过程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Navicat中怎么调试存储过程”吧!

存储过程调试方法

参考

https://www.cnblogs.com/firebata/p/4585978.html

https://www.cnblogs.com/kungfupanda/p/5645553.html

调试非常麻烦

如何在存储过程中注释?

在存储过程编辑中无法注释,所以需要先将代码拷贝到普通查询页,在查询页注释好,然后再将代码拷贝回存储过程中,保存。

变量定义

定义varchar变量必须指明大小

如varchar(10),否则会报错

游标需要定义在所有其他变量之后,否则会报 1337 错误

ERROR 1337 (42000): Variable or condition declaration after cursor

变量的定义不要和你的select的列的键同名!不然,fetch into 会失败!

这个没试,但是名字最好不要和列名同名,这样也更易读。

语法方面

存储过程中一定要记得在语句后面加上分号“;”,否则会报看起来莫名其妙的错误

调试技巧:可以使用 SELECT @a 查看打印的变量,但是注意 @a看到的是最终的变量,中间变量是无法看到的

使用 SELECT @a看到的结果中的 @a 值是最终值,中间变量是无法看到的,所以说调试非常麻烦,无法打印变量在运行中间的值

★调试的技巧

也是定位错误的方法(因为根据提示的错误很难定位到错误的语句位置):删除无关的语句,排除的方法。

边开发边测试:(1)设计好整体思路;(2)将思路框架写出来,先不要填充细节部分,比如有的复杂的实现,可以先空着或者简单的打印一些变量,边写边测,及时发现各种问题;(3)思路框架写出来后,如果测试没有问题,再补充细节代码,最好写一部分就测试一下。因为调试太麻烦的原因,这样比较稳妥。

游标的使用

参考

https://www.cnblogs.com/trying/p/3296793.html

使用

游标需要定义在所有其他变量之后,否则会报 1337 错误

如何循环?

如何结束?

如何调试?调试非常麻烦

游标使用示例存储过程1

背景说明

将表t中的数据拷贝至表t1

建表语句

-- ----------------------------

-- Table structure for t

-- ----------------------------

DROP TABLE IF EXISTS `t`;

CREATE TABLE `t` (

  `i` int(11) DEFAULT NULL

) ENGINE=InnoDB DEFAULT CHARSET=utf8;

-- ----------------------------

-- Records of t

-- ----------------------------

INSERT INTO `t` VALUES ('1');

INSERT INTO `t` VALUES ('2');

-- ----------------------------

-- Table structure for t1

-- ----------------------------

DROP TABLE IF EXISTS `t1`;

CREATE TABLE `t1` (

  `i` int(11) DEFAULT NULL

) ENGINE=InnoDB DEFAULT CHARSET=utf8;

存储过程testCopy

-- ----------------------------

-- Procedure structure for testCopy

-- ----------------------------

DROP PROCEDURE IF EXISTS `testCopy`;

DELIMITER ;;

CREATE DEFINER=`root`@`localhost` PROCEDURE `testCopy`()

BEGIN

-- 需要定义接收游标数据的变量

  DECLARE a CHAR(16);

  -- 遍历数据结束标志

  DECLARE done INT DEFAULT FALSE;

-- 游标,游标需要定义在所有其他变量之后,否则会报 1337 错误

  DECLARE cur CURSOR FOR SELECT i FROM t;

  -- 将结束标志绑定到游标

  DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;

  -- 打开游标

  OPEN cur;

  -- 开始循环

  read_loop: LOOP

    -- 提取游标里的数据,这里只有一个,多个的话也一样;

    FETCH cur INTO a;

    -- 声明结束的时候

    IF done THEN

      LEAVE read_loop;

    END IF;

    -- 这里做你想做的循环的事件

    INSERT INTO t1 VALUES (a);

  END LOOP;

  -- 关闭游标

  CLOSE cur;

END

;;

DELIMITER ;

游标使用示例存储过程2

建表语句

-- ----------------------------

-- Table structure for current

-- ----------------------------

DROP TABLE IF EXISTS `current`;

CREATE TABLE `current` (

  `player` varchar(11) DEFAULT NULL,

  `market` varchar(11) DEFAULT NULL,

  `money` int(11) DEFAULT NULL,

  `dateno` int(11) DEFAULT NULL

) ENGINE=InnoDB DEFAULT CHARSET=utf8;

-- ----------------------------

-- Records of current

-- ----------------------------

INSERT INTO `current` VALUES ('小明', '永辉', '100', '20190807');

INSERT INTO `current` VALUES ('小明', '物美', '200', '20190807');

INSERT INTO `current` VALUES ('小刚', '家乐福', '230', '20190807');

-- ----------------------------

-- Table structure for history

-- ----------------------------

DROP TABLE IF EXISTS `history`;

CREATE TABLE `history` (

  `player` varchar(11) DEFAULT NULL,

  `market` varchar(11) DEFAULT NULL,

  `money` int(11) DEFAULT NULL,

  `dateno` int(11) DEFAULT NULL

) ENGINE=InnoDB DEFAULT CHARSET=utf8;

-- ----------------------------

-- Records of history

-- ----------------------------

背景说明

表current,表history

结构一致,均为

player 送货员

market 超市

money 送货钱数

dateno 日期序号

想象一个送货员可以帮多家超市(永辉、物美、家乐福等)送货,每次送货都会是一定的金额(从几十元到几百上千元)。

current记录的是今天某个送货员给给家送货的金额情况。

history记录的今天之前的累计情况。

那么存储过程testMerge就是执行这么个操作:将current表中的数据累计到history表中,然后删除累计过的数据。

思路:(1)选取current表中符合条件的记录到游标;(2)操作游标,看是往history表中插入记录,还是更新history表中的记录;(3)操作完毕关闭游标。

往history表中插入记录,还是更新history表中的记录的判断条件:使用【送货员,超市】联合键判断,history中存在【送货员,超市】联合键,则将current中的送货钱数累加到之前的记录;history中不存在【送货员,超市】联合键,则向history中新增一条记录。

暂时只是测试例子,实际可继续完善。

存储过程testMerge

-- ----------------------------

-- Procedure structure for testMerge

-- ----------------------------

DROP PROCEDURE IF EXISTS `testMerge`;

DELIMITER ;;

CREATE DEFINER=`root`@`localhost` PROCEDURE `testMerge`()

BEGIN

-- 需要定义接收游标数据的变量

  DECLARE player_param VARCHAR(10);

DECLARE market_param VARCHAR(10);

DECLARE money_param INT;

DECLARE exist_record INT;

  -- 遍历数据结束标志

  DECLARE done INT DEFAULT FALSE;

-- 游标,游标需要定义在所有其他变量之后,否则会报 1337 错误

  DECLARE record_list CURSOR FOR

SELECT

player,

market,

money

FROM

current

WHERE

dateno < 20190808;

  -- 将结束标志绑定到游标

  DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;

  -- 打开游标

  OPEN record_list;

  -- 开始循环

  read_loop: LOOP

    -- 提取游标里的数据,这里只有一个,多个的话也一样;

    FETCH record_list INTO player_param,market_param,money_param;

    -- 声明结束的时候

    IF done THEN

      LEAVE read_loop;

    END IF;

SELECT COUNT(*) INTO @exist_record FROM history WHERE player = player_param AND market = market_param;

IF @exist_record = 0 THEN

    -- 这里做你想做的循环的事件

INSERT INTO history VALUES (player_param,market_param,money_param,0);

END IF;

IF @exist_record > 0 THEN

UPDATE history SET money = money + money_param WHERE player = player_param AND market = market_param;

END IF;

DELETE FROM current WHERE dateno = 20190807;

  END LOOP;

  -- 关闭游标

  CLOSE record_list;

END

;;

DELIMITER ;

感谢各位的阅读,以上就是“Navicat中怎么调试存储过程”的内容了,经过本文的学习后,相信大家对Navicat中怎么调试存储过程这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是本站,小编将为大家推送更多相关知识点的文章,欢迎关注!

《Navicat中怎么调试存储过程.doc》

下载本文的Word格式文档,以方便收藏与打印。