Mysql存储过程优化——使用临时表代替游标(转)

本文通过测试MySQL游标处理不同数据量的表现,发现游标在处理小数据量时效率较高,但在处理超过一万条记录的大数据量时效率大幅下降,甚至出现错误。文章对比了使用游标与临时表的方法,并提供了具体的存储过程示例。

Mysql游标在操作小数据量时比较方便,效率可观,但操作大数据量,速度比较慢,甚至直接产生系统错误。

一般说来,当操作的数据超过1万条时,就避免用游标吧。

为了测试游标性能,写了下面一个游标对IDC_Gather_Info表中数据进行遍历

 

View Code
CREATE DEFINER=`root`@`%` PROCEDURE `debug`(IN `beginTime` int, IN `checkTime` int)  
BEGIN  
    DECLARE t_id VARCHAR(64) DEFAULT '';  
    DECLARE t_item TINYINT DEFAULT 0;  
    DECLARE t_result VARCHAR(8192) DEFAULT '';  
  
    DECLARE cursorDone INT DEFAULT 0;  
    DECLARE cur CURSOR FOR SELECT Asset_Id, Check_Item, Check_Result from IDC_Gather_Info WHERE Check_Time > beginTime AND Check_Time <= checkTime;  
    DECLARE CONTINUE HANDLER FOR SQLSTATE '02000' SET cursorDone = 1;  
  
    OPEN cur;  
    cursorLoop:LOOP  
        FETCH cur INTO t_id, t_item, t_result;  
        IF cursorDone = 1 THEN  
            LEAVE cursorLoop;  
        END IF;  
    END LOOP;  
    CLOSE cur;  
END

 

1.数据量15万,存储过程执行失败,提示错误:Incorrect key file for table '/tmp/#sql_3044_0.MYI';try to repair it

2.数据量5万,执行成功,耗时31.051s

3.数据量1万,执行成功,耗时1.371s

下面使用临时表替换游标:1.数据量15万,执行成功,耗时8.928s

View Code
CREATE DEFINER=`root`@`%` PROCEDURE `debug`(IN `beginTime` int, IN `checkTime` int)  
    BEGIN  
        DECLARE t_id VARCHAR(64) DEFAULT '';  
        DECLARE t_item TINYINT DEFAULT 0;  
        DECLARE t_result VARCHAR(8192) DEFAULT '';  
          
        DECLARE maxCnt INT DEFAULT 0;  
        DECLARE i INT DEFAULT 0;  
      
        DROP TABLE IF EXISTS Gather_Data_Tmp;  
        CREATE TEMPORARY TABLE Gather_Data_Tmp(  
            `Tmp_Id` INT UNSIGNED NOT NULL AUTO_INCREMENT,  
            `Asset_Id` VARCHAR(16) NOT NULL,  
            `Check_Item` TINYINT(1) NOT NULL,  
            `Check_Result` VARCHAR(8192) NOT NULL,  
            PRIMARY KEY (`Tmp_Id`)  
        )ENGINE=MyISAM DEFAULT CHARSET=utf8;  
      
        SET @tSql = CONCAT('INSERT INTO Gather_Data_Tmp (`Asset_Id`, `Check_Item`, `Check_Result`)   
                                                SELECT Asset_Id, Check_Item, Check_Result   
                                                FROM IDC_Gather_Info   
                                                WHERE Check_Time > ',beginTime,' AND Check_Time <= ',checkTime);  
        PREPARE gatherData FROM @tSql;  
        EXECUTE gatherData;  
      
        SELECT MIN(`Tmp_Id`) INTO i FROM Gather_Data_Tmp;  
        SELECT MAX(`Tmp_Id`) INTO maxCnt FROM Gather_Data_Tmp;  
      
        WHILE i <= maxCnt DO  
            SELECT Asset_Id, Check_Item, Check_Result INTO t_id, t_item, t_result FROM Gather_Data_Tmp WHERE Tmp_Id = i;  
            SET i = i + 1;  
        END WHILE;  
    END

2.数据量5万,执行成功,耗时2.994s
3.数据量1万,执行成功,耗时0.634s


可以看到Mysql的游标在处理大一点的数据量时还是比较乏力的,仅适合用于操作几百上千的小数据量。

 

转载于:https://www.cnblogs.com/tonykan/archive/2012/11/28/2793807.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值