IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> Mysql过长的Sql语句优化:在循环中调用存储过程运行动态语句 -> 正文阅读

[大数据]Mysql过长的Sql语句优化:在循环中调用存储过程运行动态语句

????????在批量查询的时候,有时为了提高检索效率,会使用Unique配合IN查询。此时注意IN中不要写子查询,否则时间复杂度会大大增加。不过对于已经确定的检索范围,如果有Unique索引,效率还是可以的。

????????但有时候由于种种原因,并不总会有Unique索引。那么当表的数据量比较大(达到千万级)的时候,查询的时间就比较长了。因为IN会进行全表搜索。而且如果想去重的话,那就要配合Group By语句了。

????????已知的改进的方案有若干种:①用OR取代IN②在WHERE子句中限定IN查询范围的上下限,③用UNION的方式合并查询结果,等等。

? ? ? ? 如果单纯用UNION执行单次查询的话,查询的效率通常还是会受到影响。它还是会进行全表检索,而且会检索多次。而且,如果是单次检索,可能会构造出行数很多的SQL语句,MYSQL服务器解析SQL语句并处理的时候也会慢下来。

????????下面这种改进方案是基于查询范围上限进行的。

????????①在应用程序中(如JVM),将查询范围的集合元素按升(或降,后面注意相应调整)序排序。

? ? ? ? ②在MYSQL中建立两个函数:func_get_split_string,用来截取字符串;func_get_split_string_total,获取被分隔符(如',')分隔的查询元素的总数。如下:

-- 截取字符串
DROP FUNCTION IF EXISTS func_get_split_string;
DELIMITER $$
CREATE DEFINER=`root`@`%` FUNCTION `func_get_split_string`(
f_string varchar(16382),f_delimiter varchar(5),f_order int) RETURNS varchar(255) CHARSET utf8
BEGIN
 -- Get the separated number of given string.
 declare result varchar(255) default '';
 set result = reverse(substring_index(reverse(substring_index(f_string,f_delimiter,f_order)),f_delimiter,1));
 return result;
END$$
DELIMITER ;


-- 元素总数
DROP FUNCTION IF EXISTS func_get_split_string_total;
DELIMITER $$
CREATE DEFINER=`root`@`%` FUNCTION `func_get_split_string_total`(
f_string varchar(16382),f_delimiter varchar(5)
) RETURNS int(11)
BEGIN
 -- Get the total number of given string.
 return 1+(length(f_string) - length(replace(f_string,f_delimiter,'')));
END$$
DELIMITER ;

????????②写一个存储过程,入参为IN的结果集用一个特定的字符拼接而成的(因为Mysql的存储过程不支持集合)。如果搜索集元素较多或较长,参数类型应设为TEXT或LONGTEXT。这是入口存储过程。

? ? ? ? ④在入口存储过程中,用函数func_get_split_string拆解输入的范围的元素,然后在循环中调用执行子任务的存储过程。

????????

DROP PROCEDURE IF EXISTS Pro_entry;
DELIMITER $$
CREATE PROCEDURE Pro_entry(
    _cIds TEXT) -- 定义参数
BEGIN
    DECLARE cnt int default 0;
    DECLARE i int default 0;
    DECLARE _cId VARCHAR(128);//当前元素
    DECLARE _priorCid VARCHAR(128) DEFAULT NULL;//前一元素

    SET cnt = func_get_split_string_total(_cIds,',');//获得元素个数
    while i < cnt
    do
        set i = i + 1;
        SET _cId = func_get_split_string(_cIds,',',i);
        CALL Proc_sub(_cId, _priorCid);//调用子存储过程进行查询操作
    END WHILE;

END$$
DELIMITER ;

? ? ? ? ⑤定义执行存储过程Proc_sub,可用Limit限制每次的检索行数。注意用到动态指令PREPARE、EXECUTE执行:????????

DROP PROCEDURE IF EXISTS Pro_sub;
DELIMITER $$
CREATE PROCEDURE Pro_sub (
    INOUT annualAssetsSql LONGTEXT,
    IN _cId VARCHAR ( 128 ),
    IN _priorCid VARCHAR ( 128 )
)
BEGIN -- 按照Sql的逻辑层次排布代码
    DECLARE thisSql LONGTEXT;

    SET thisSql = CONCAT(
        ' SELECT * FROM 【表名】 tb
        WHERE tb.c_id = ''', _cId, ''' '
    );
    IF _priorCid IS NOT NULL THEN
        SET thisSql = CONCAT(thisSql, ' AND tb.c_id > ''', _priorCid, ''' '); 
    END IF;
    SET thisSql = CONCAT(thisSql, ' LIMIT 1 )');

    set @annual_assets_sql = thisSql;
    PREPARE testSelectTemp FROM @annual_assets_sql;
    EXECUTE testSelectTemp;
    DEALLOCATE PREPARE testSelectTemp;
END$$
DELIMITER ;

????????其实,更推荐的方式是方法②,即在确定了上下限的情况下配合IN使用。但这也是会产生较大的SQL语句。

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章      下一篇文章      查看所有文章
加:2022-03-17 22:14:33  更:2022-03-17 22:17:30 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/16 18:00:01-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码