如何使用预准备语句(MySQL)动态DROP存储过程和函数 - DROP PROCEDURE LIKE

时间:2016-03-17 19:08:11

标签: mysql stored-procedures mysql-workbench sql-drop

我一直在尝试动态删除MySQL中的表,过程和函数。我这样做是因为我为项目动态创建它们,当项目版本发生变化时我需要清理并重建它。

可以 动态删除;我 无法 动态删除程序功能

以下是我正在使用的代码示例:

DELIMITER ;;
DROP PROCEDURE IF EXISTS md_remove_project; ;;
CREATE PROCEDURE         md_remove_project()
begin
    DECLARE TableName text;
    DECLARE ProcName  text;
    DECLARE done      int DEFAULT false;
    DECLARE statement text;

    DECLARE table_cursor CURSOR FOR
        SELECT table_name FROM tmp_md_tables;

    DECLARE proc_cursor CURSOR FOR
        SELECT routine_name FROM tmp_md_procedures;

    DECLARE func_cursor CURSOR FOR
        SELECT routine_name FROM tmp_md_functions;

    DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = true;       

    # Drop all the 'md' tables..............................................
    # This Works...
    DROP TABLE IF EXISTS   tmp_md_tables;
    CREATE TEMPORARY TABLE tmp_md_tables
    SELECT
        table_name
    FROM
        information_schema.tables
    WHERE
        table_name LIKE 'md_%';

    OPEN table_cursor;
    table_loop: LOOP
        FETCH table_cursor INTO TableName;
        IF done THEN
            LEAVE table_loop;
        END IF;

        SET @statement = CONCAT('DROP TABLE IF EXISTS ', TableName, ';');
        PREPARE STATEMENT FROM @statement;
        EXECUTE STATEMENT;
        DEALLOCATE PREPARE STATEMENT;        
    END LOOP;
    CLOSE table_cursor;
    DROP TABLE IF EXISTS tmp_md_tables;
    #-----------------------------------------------------------------------
    # Drop all the 'md' procedures............................................
    DROP TABLE IF EXISTS   tmp_md_procedures;
    CREATE TEMPORARY TABLE tmp_md_procedures
    SELECT
        routine_name
    FROM
        information_schema.routines
    WHERE
        routine_type = 'PROCEDURE'
        and
        routine_name LIKE 'md_%';

    SET done = false;
    OPEN proc_cursor;
    proc_loop: LOOP
        FETCH proc_cursor INTO ProcName;

        IF ProcName = 'md_remove_project' THEN
            ITERATE proc_loop;
        END IF;

        IF done THEN
            leave proc_loop;
        END IF;

        SET @statement = CONCAT('DROP PROCEDURE IF EXISTS ', ProcName, ';');
        PREPARE STATEMENT FROM @statement;
        EXECUTE STATEMENT;
        DEALLOCATE PREPARE STATEMENT;
    END LOOP;
    CLOSE proc_cursor;
    DROP TABLE IF EXISTS tmp_md_procedures;
END;
;;
DELIMITER ;

#CALL md_remove_project;

所以我用一个名为md_%的过程创建一个表,然后循环遍历表。对于每个routine_name,我准备一个语句来删除该过程。然后我收到以下消息:

错误代码:1295。准备好的语句协议中不支持此命令

有没有其他解决方案可以放弃像“md_%”这样的程序???

谢谢。

1 个答案:

答案 0 :(得分:0)

使用mysqli_...函数时,无需尝试更改分隔符。只有在使用MySQL(命令行)客户端时才需要使用change delimiter命令。事实上,该命令是一个MySQL客户端命令(客户端从不将其发送到服务器)。

服务器非常智能,能够识别CREATE PROCEDURE命令,并知道它以END;

结束

因此,您只需执行两项查询:首先是DROP PROCEDURE IF EXISTS ...,然后是CREATE PROCEDURE ... END;查询。

如果您必须在一次通话中执行这些操作,您可以使用mysqli::multi_query但我会建议不要这样做(因为可能会产生严重的安全隐患)。