用于预处理语句的MySQL存储过程游标

时间:2014-07-30 20:56:45

标签: mysql stored-procedures

我有两张桌子:

people_en: id, name
people_es: id, name

(拜托,请不要理会规范化。设计是规范化的。表格要比这复杂得多 但这只是一种简化我的问题的方法。)

然后我有一个存储过程:

CREATE PROCEDURE myproc(lang char(2))
BEGIN
    set @select = concat('SELECT * FROM ', lang, ' limit 3');
    PREPARE stm FROM @select;
    EXECUTE stm;
    DEALLOCATE PREPARE stm;
    SET @cnt = FOUND_ROWS(); 
    SELECT @cnt;
    IF @cnt = 3 THEN
        //Here I need to loop through the rows
    ELSE
        //Do something else
    END IF;
   END$$

或多或少,程序中的逻辑是:

如果select给出3行,那么我们必须遍历行并使用每行中的值做一些事情。 否则另外做什么(不重要的是什么,但是我这样说是为了让你理解我需要的东西 循环前的if语句。

我已经看过并阅读过关于游标的信息,但是找不到由concat创建的选项(这有关系吗?) 特别是用准备好的声明创建。 如何遍历结果列表并使用每行的值? 感谢。

2 个答案:

答案 0 :(得分:15)

我有一些坏消息和好消息。

首先是坏消息。

  

MySQL手册说游标不能用于动态语句   准备并执行PREPARE和EXECUTE。一份声明   在光标创建时检查游标,因此语句不能   动态的。

所以到目前为止还没有动态游标......在这里你需要这样的东西。

但现在好消息:至少有两种方法可以绕过它 - 使用vw或tbl。

下面我重新编写了您的代码并应用了视图来制作动态的'光标。

DELIMITER //

DROP PROCEDURE IF EXISTS myproc;
CREATE PROCEDURE myproc(IN lang VARCHAR(400))

BEGIN

    DECLARE c VARCHAR(400);
    DECLARE done BOOLEAN DEFAULT FALSE;
    DECLARE cur CURSOR FOR SELECT name FROM vw_myproc;
    DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;

    SET @select = concat('CREATE VIEW vw_myproc as SELECT * FROM ', lang, ' limit 3');
    PREPARE stm FROM @select;
    EXECUTE stm;
    DEALLOCATE PREPARE stm;

    SET @select = concat('SELECT * FROM ', lang, ' limit 3');
    PREPARE stm FROM @select;
    EXECUTE stm;
    DEALLOCATE PREPARE stm;

    SET @cnt = FOUND_ROWS(); 
    SELECT @cnt;
    IF @cnt = 3 THEN
          OPEN cur;
          read_loop: LOOP
            FETCH cur INTO c;
            IF done THEN
              LEAVE read_loop;
            END IF;

            #HERE YOU CAN DO STH WITH EACH ROW e.g. UPDATE; INSERT; DELETE etc
            SELECT c;

          END LOOP read_loop;
          CLOSE cur;
          DROP VIEW vw_myproc;
    ELSE
        SET c = '';
    END IF;

END//

DELIMITER ;

测试程序:

CALL myproc('people_en');

答案 1 :(得分:5)

@clickstefan,您将遇到两个或更多用户同时尝试执行脚本的问题。第二个用户将收到错误消息' View vw_myproc已经存在'对于该行:

SET @select = concat('CREATE VIEW vw_myproc as SELECT * FROM ', lang, ' limit 3');

解决方案是临时表 - 它仅存在于当前连接的生命周期中,用户可以同时创建具有相同名称的临时表。因此,代码可能如下所示:

DROP TABLE IF EXISTS vw_myproc;
SET @select = concat('CREATE TEMPORARY TABLE vw_myproc AS SELECT * FROM ', lang, ' limit 3');