• 菜鸟使用MySQL存储过程and临时表,供新手参考,请高手斧正


      因为公司最近的一个项目,第一次用到了MySQL(5.10版本),之前听传说MySQL很厉害的样子,因为开源而神奇,但是现在用起来,

    感觉并不好啊!我知道是我水平太down,呜呜呜,请各路神仙略施小技,拯救我于水深火热之中。

      本文主要针对以下两个问题:

      (1)存储过程单个参数传入值集合。

      (2)使用临时表实现实现对同一结果集的多次使用。

      也不说太多废话了,有下面数据表tb_address和数据,各个地区通过id和parentid实现层级关系:

     

      还有一个表tb_population,记录了那个区有多少人口,id是主键,townid是区的id:

     功能一: 

      现在要实现的功能是通过地区id查询人口,可以一次查询多个地区。为了实现这个功能,我创建了一个存储过程,

    但是怎么传入多个id呢?这个难倒了我,在网上找了不少方法,终于用下面的代码实现了:

    DROP PROCEDURE IF EXISTS pro_getDownPopulation;
    CREATE PROCEDURE pro_getDownPopulation(ids BLOB)
    BEGIN 
    SET @str1 = 'SELECT a.id,
        a.address, 
        b.population 
    from tb_address a 
    RIGHT JOIN tb_population b on a.id=b.townid WHERE a.id in (';
    SET @str2 = ')';
    SET @asql = CONCAT(@str1, ids, @str2);
    
    PREPARE stmt FROM @asql;
    EXECUTE stmt;
    DEALLOCATE PREPARE stmt;
    END;

    执行该存储过程pro_getDownPopulation,得到结果如下:

    实现是实现了,总感觉有点怪怪的,后来继续开发的过程中无意间发现有个FIND_IN_SET()函数,果断用上了,就是不知道性能好不好,

    修改后的存储过程代码如下:

    DROP PROCEDURE IF EXISTS pro_getDownPopulation2;
    CREATE PROCEDURE pro_getDownPopulation2(ids BLOB)
    BEGIN 
            SELECT a.id,
                a.address, 
                b.population 
            from tb_address a 
            RIGHT JOIN tb_population b on a.id=b.townid WHERE FIND_IN_SET(a.id, ids);
    END;

    执行该存储过程pro_getDownPopulation2,得到结果如下:(注意两次传入参数的格式的不同)

    我的两种实现方式就是这样,到底哪种好,没有经过性能方面的测试,还请大家指教!

    功能二:

       按地方的层级(省、市、区)统计个地方的人口,并以树的结构显示统计结果(软件端显示)。因为有一个简单的结果集我要反复用到,

    所以我使用了临时表,但是在网上查到的说法:同一个query里同一个临时表只能使用一次,故创建了多个相同结果的临时表,代码如下:

    DROP PROCEDURE IF EXISTS pro_population_statistics;
    CREATE PROCEDURE pro_population_statistics()
    BEGIN 
    DROP TEMPORARY TABLE IF EXISTS tempTable1;
    DROP TEMPORARY TABLE IF EXISTS tempTable2;
    DROP TEMPORARY TABLE IF EXISTS tempTable3;
    
    CREATE TEMPORARY TABLE tempTable1
    SELECT a.id shengID, a.address sheng,
            b.id shiID, b.address shi,
            c.id quID, c.address qu, 
            d.population FROM tb_address a
    LEFT JOIN tb_address b on a.id = b.parentid
    LEFT JOIN tb_address c on b.id = c.parentid
    LEFT JOIN tb_population d on c.id = d.townid
    WHERE a.parentid='';
    
    CREATE TEMPORARY TABLE tempTable2 SELECT * from tempTable1;
    CREATE TEMPORARY TABLE tempTable3 SELECT * from tempTable1;
    
    SELECT sheng, SUM(population) population from tempTable1 GROUP BY shengID
    UNION
    SELECT shi, SUM(population) from tempTable2 GROUP BY shiID
    UNION
    SELECT qu, population from tempTable3;
    
    END;

    执行存储过程pro_population_statistics,结果如下:

      

      功能是实现了,但是我存在很多疑惑的地方,特别是同一个query里同一个临时表只能使用一次,

    我使用多个临时表,用起来性能比较差,大家有什么解决的方法吗?

      我还有另外一个疑问:在一个存储过程里如何调用另外一个存储过程并保存后者返回的结果集?

    很少写文章,排版上的问题请大家多多包涵and多多指导,谢谢!

  • 相关阅读:
    enca工具,检测文件编码
    ubuntu 支持gbk
    LinkedList线程安全问题
    php防止form重复提交的方法
    Linux 内存泄露调试工具
    从B树、B+树、B*树谈到R 树
    Ubuntu 语言设置
    wwwauthenticate
    Lua脚本语法说明
    jQuery学习总结之元素的相对定位和选择器持续更新中
  • 原文地址:https://www.cnblogs.com/cqpass/p/cqpass_ed.html
Copyright © 2020-2023  润新知