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 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> 接口响应优化,sql优化实战 -> 正文阅读

[大数据]接口响应优化,sql优化实战

接口响应速度达到20s

前端展示接口平均速度为8s,最坏情况到达20s
在这里插入图片描述
默认搜索条件十分复杂,基本每个字段都传条件,不好建立联合索引
在这里插入图片描述

多线程异步查询

接口内容分为两部分,并且查询相互独立
1、查询页面集合(方法内存在很多业务逻辑,不是简单查询)
2、查询总数 (方法内存在很多业务逻辑,不是简单查询)
在这里插入图片描述所以使用线程池来执行查询操作,这里使用Callable接口,Callable对象执行后可以有返回值,运行Callable任务可以得到一个Future对象,通过Future对象可以了解任务执行情况,可以使用future.get()方法等待线程的执行结果。(这里也可以使用CountDownLanch)

在这里插入图片描述

成功解决?

本以为到这里简单处理一下,就可以达到需求方可以接受的层度,自测接口一看
线程1耗时:6000ms
线程2耗时:7000ms
没办法只有梳理上百行的业务和sql(苦逼)!!!
一条一条的找到慢sql(这里还是精简后的sql,实际查询条件那叫一个眼花缭乱)

这里命名为:慢sqlA,执行耗时7s

SELECT COUNT(1) FROM ( 
SELECT COUNT(1) FROM `tb_product_information_pass` AS pass LEFT JOIN tb_product_contacts 
AS con ON con.product_information_id = pass.id LEFT JOIN tb_product_query_data pqd ON 
pqd.product_id = pass.id WHERE `product_status` = 1 AND pass.`show_id` != 1
 GROUP BY product_group_name ) AS t 

这里命名为:慢sqlB,执行耗时5s

SELECT COUNT(DISTINCT pass.id) AS `all`, 
SUM(IF(pass.`customer_rank` = '003003001', 1, 0)) AS `hide`, 
SUM(IF(pass.`customer_rank` = '003003002', 1, 0)) AS `toContact`, 
SUM(IF(pass.`customer_rank` = '003003003', 1, 0)) AS `setContact`, 
SUM(IF(pass.`customer_rank` = '003003004', 1, 0)) AS `cooperationed`, 
SUM(IF(pass.`customer_rank` = '003003005', 1, 0)) AS `lose` 
FROM ( 
 SELECT pc_index_data,pass.*
FROM `tb_product_information_pass` AS pass
LEFT JOIN tb_product_query_data pqd ON pqd.product_id = pass.id  
 GROUP BY pass.id 
) AS pass   
LEFT JOIN tb_product_contacts AS con ON con.product_information_id = pass.id 
WHERE  pass.`show_id` != 1 and `business_affairs` IN ( '方' ) and pass.`product_status` = 1

这里命名为:慢sqlC,耗时6s

SELECT pc_index_data, COUNT(temp.productId) AS child_amount,`productId`, `product_name`, `product_source`, `product_iscompany`, `company_name`, `company_nature`, `user_name`, 
`id_card`, `contact_address`, `customer_level`, `control_volume`, `number`, `settlement_cycles`, `settlement_method`, `share_proportion`, `cycle`, `kpi`, `advance_payment`, 
`payment_method`, `business_affairs`, `not_cooperation_reasons`, `loss_feedback`, `input_time`, `customer_rank`, `related_Id`, `edit_type`, `delete_type`, `second_id`, `person_id`, 
`productPass`, `product_status`, `contract_time`, `second_signing_people`, `second_company_term`, `contract_code`, `stage_time` , `hits` , `contact_number`,`product_group_name` 
FROM ( 
  SELECT MAX( IFNULL(pqd.pc_index_data,0)) AS pc_index_data, pass.`id` AS `productId`, `product_name`, `product_source`, `product_iscompany`, `company_name`, `company_nature`, `user_name`, `id_card`,
   `contact_address`, `customer_level`, `control_volume`, `number`, `settlement_cycles`, `settlement_method`, `share_proportion`, `cycle`, `kpi`, `advance_payment`, `payment_method`, `business_affairs`, 
   `not_cooperation_reasons`, `loss_feedback`, `input_time`, `customer_rank`, `related_Id`, `edit_type`, `delete_type`, `second_id`, `person_id`, pass.`pass_type` AS `productPass`, `product_status`,
    `contract_time`, `second_signing_people`, `second_company_term`, `contract_code`, `stage_time` , `hits` , '' AS `contact_number`,`product_group_name` 
    FROM `tb_product_information_pass` AS pass LEFT JOIN tb_product_contacts AS con ON con.product_information_id = pass.id LEFT JOIN tb_product_query_data pqd ON pqd.product_id = pass.id 
    WHERE `product_status` = 1 AND pass.`show_id` != 1 GROUP BY pass.id ) AS temp GROUP BY product_group_name ORDER BY `productPass` ASC, input_time DESC 

数据量情况
tb_product_information_pass:7w
tb_product_query_data :300w
tb_product_contacts :8w

慢sqlA优化

我心伤悲

对product_status和product_group_name字段建立联合索引,查询速度并没有明显变化。
多表联查优化策略:
1、left join时候采用小表驱动大表的方式,即把数据量较小的表放在左边(无明显)
==2、将left join 改成inner join(实际上是减少数据量,但是这里右表的null值对业务来说任然具有意义) ==
在这里插入图片描述这里用了出现了using filesort与using temporate,using index,using where。
1、using index:当我们的查询列表以及搜索条件中只包含属于某个索引的列, 也就是在可以使用索引覆盖的情况下会出现
2、using where:当我们使用全表扫描来执行对某个表的查询, 并且该语句的 WHERE 子句中有针对该表的搜索条件时, 在 Extra 列中会提示上述额外信息。当使用索引访问来执行对某个表的查询, 并且该语句的 WHERE 子句中有除了该索引包含的列之外的其他搜索条件时, 在 Extra 列中也会提示上述信息。
3、using filesort:多出现在分组或者排序时,很多情况下排序操作无法使用到索引, 只能在内存中(记录较少的时候) 或者磁盘中(记录较多的时候) 进行排序, MySQL把这种在内存中或者磁盘上进行排序的方式统称为文件排序。 如果某个查询需要使用文件排序的方式执行查询, 就会在执行计划的 Extra 列中显示 Using filesort提示
4、using temporate:在许多查询的执行过程中, MySQL 可能会借助临时表来完成一些功能, 比如去重、 排序之类的, 比如我们在执行许多包含 DISTINCT、 GROUP BY、 UNION 等子句的查询过程中, 如果不能有效利用索引来完成查询, MySQL 很有可能寻求通过建立内部的临时表来执行查询。 如果查询中使用到了内部的临时表, 在执行计划的 Extra 列将会显示 Using temporary 提示

经过测试发现,这里的sql主要是由于group by而变慢的
group by 实际上等于group by column order by column,所以将
GROUP BY product_group_name 改为直接排序
GROUP BY product_group_name order by null,但是sql并没有明显提速。

柳暗花明又一村

优化using temporate
mysql在使用临时表时会首先使用内存临时表,当数据量溢出内存临时表时切换磁盘临时表
SQL_BIG_RESULT:这个提示告诉mysql直接走磁盘临时表,跳过内存临时表。
惊了(没干什么都提速到毫秒)
优化后的sql:

SELECT COUNT(1) FROM ( SELECT SQL_BIG_RESULT COUNT(1) FROM `tb_product_information_pass` AS pass LEFT JOIN tb_product_contacts AS con ON con.product_information_id = pass.id LEFT JOIN tb_product_query_data pqd ON pqd.product_id = pass.id WHERE `product_status` = 1 AND pass.`show_id` != 1 GROUP BY product_group_name ) AS t 

慢sqlB优化

这里比较简单,前置搜索条件减少子查询的数据量,sql响应速度从5s变为1s。(nice)

SELECT COUNT(DISTINCT pass.id) AS `all`, 
SUM(IF(pass.`customer_rank` = '003003001', 1, 0)) AS `hide`, 
SUM(IF(pass.`customer_rank` = '003003002', 1, 0)) AS `toContact`, 
SUM(IF(pass.`customer_rank` = '003003003', 1, 0)) AS `setContact`, 
SUM(IF(pass.`customer_rank` = '003003004', 1, 0)) AS `cooperationed`, 
SUM(IF(pass.`customer_rank` = '003003005', 1, 0)) AS `lose` 
FROM ( 
 SELECT pc_index_data,pass.*
FROM `tb_product_information_pass` AS pass
LEFT JOIN tb_product_query_data pqd ON pqd.product_id = pass.id  
WHERE `business_affairs` IN ( '方' ) and pass.`product_status` = 1 GROUP BY pass.id 
) AS pass   
LEFT JOIN tb_product_contacts AS con ON con.product_information_id = pass.id 
WHERE  pass.`show_id` != 1 

继续努力慢sqlC优化

分解sql

这条sql两层分组,每层分组都具有聚合函数,拆解sql,其主要耗时再MAX( IFNULL(pqd.pc_index_data,0))这条函数上,其数据量100w条,在这个基本上载进行函数操作可想而知的慢。
这里我首先想到的是有没有办法从sql层面削去两层分组和函数。(并没有找到合适的方法,跪求sql语句层面的优化)
由于数据量的问题,内存中处理函数也不可取

分析业务和数据表。

tb_product_query_data 表的数据量达到惊人的300w,但是联表查询只用了一个字段
pc_index_data,是不是可以减少一个连表查询内,在tb_product_information_pass表建立冗余字段pc_index_data,降低查询数据量呢

说干就干,建冗余字段

1、建冗余字段

在这里插入图片描述

2、并将tb_product_query_data 的数据同步置tb_product_information_pass

UPDATE tb_product_information_pass a , (SELECT product_id,MAX(IFNULL(pc_index_data,0)) AS pc_index_data FROM tb_product_query_data GROUP BY product_id) b SET a.pc_index_data = b.pc_index_data WHERE a.id = b.`product_id`

3、建立触发器,监控tb_product_query_data 表的数据变化同步到tb_product_information_pass

DELIMITER $$

USE `bz_zone_test`$$

DROP TRIGGER /*!50032 IF EXISTS */ `insert_product_query`$$

CREATE
    /*!50017 DEFINER = 'bz_zone_test'@'%' */
    TRIGGER `insert_product_query` AFTER INSERT ON `tb_product_query_data` 
    FOR EACH ROW BEGIN
UPDATE tb_product_information_pass SET pc_index_data = IF(new.pc_index_data <= pc_index_data,pc_index_data,new.pc_index_data)
WHERE id = new.product_id;
END;
$$

DELIMITER ;
DELIMITER $$

USE `bz_zone_test`$$

DROP TRIGGER /*!50032 IF EXISTS */ `update_product_query`$$

CREATE
    /*!50017 DEFINER = 'bz_zone_test'@'%' */
    TRIGGER `update_product_query` AFTER UPDATE ON `tb_product_query_data` 
    FOR EACH ROW BEGIN
UPDATE tb_product_information_pass SET pc_index_data = IF(new.pc_index_data <= pc_index_data,pc_index_data,new.pc_index_data)
WHERE id = new.product_id;
END;
$$

DELIMITER ;

修改原sql,去掉烦人的函数

速度优化置1s内(开心)

大功告成

成就感十足
在这里插入图片描述

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章      下一篇文章      查看所有文章
加:2022-06-06 17:25:11  更:2022-06-06 17:27:16 
 
开发: 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 4:49:49-

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