博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
美团点评开源 SQL 优化工具 SQLAdvisor 测试报告
阅读量:6029 次
发布时间:2019-06-20

本文共 7076 字,大约阅读时间需要 23 分钟。

hot3.png

一、SQLAdvisor 介绍

SQLAdvisor 是由美团点评公司北京 DBA 团队开发维护的 SQL 优化工具:输入 SQL,输出索引优化建议。它基于 MySQL 原生词法解析,再结合 SQL 中的 where 条件以及字段选择度、聚合条件、多表 Join 关系等最终输出最优的索引优化建议。目前 SQLAdvisor 在美团公司内部大量使用,较为成熟、稳定。

SQLAdvisor 的优点

  • 基于 MySQL 原生词法解析,充分保证词法解析的性能、准确定以及稳定性;
  • 支持常见的 SQL(Insert/Delete/Update/Select);
  • 支持多表 Join 并自动逻辑选定驱动表;
  • 支持聚合条件 Order by 和 Group by;
  • 过滤表中已存在的索引。

 

二、SQLAdvisor 原理

SQLAdvisor 架构流程图:

171251_Y2q5_3023401.png

SQLAdvisor 包含了如下的处理方式:Join 处理、where 处理、计算区分度、添加备选索引、Group 与 Order 处理、驱动表选择、添加被驱动表备选索引、输出建议,具体的流程图可以参考:

 

三、SQLAdvisor 测试

3.1  SQLAdvisor 安装

3.1.1 拉取最新代码git clone https://github.com/Meituan-Dianping/SQLAdvisor.git3.1.2 安装依赖项yum install -y cmake libaio-devel libffi-devel glib2 glib2-devel bison# 因 yum 安装 Percona-Server-shared-56 失败,故使用 rpm 包安装,\# 具体参考 https://github.com/Meituan-Dianping/SQLAdvisor/issues/12yum install -y --enablerepo=Percona56 Percona-Server-shared-56yum install -y Percona-Server-server-56 Percona-Server-client-56rpm -ivh Percona-Server-shared-56-5.6.25-rel73.1.el6.x86_64.rpm# 设置软链cd /usr/lib64/ls -l libperconaserverclient_r.so.18ln -s libperconaserverclient_r.so.18 libperconaserverclient_r.so3.1.3 编译依赖项 sqlparsercmake -DBUILD_CONFIG=mysql_release -DCMAKE_BUILD_TYPE=debug \-DCMAKE_INSTALL_PREFIX=/usr/local/sqlparser ./make && make install3.1.4 安装 SQLAdvisor 源码cd sqladvisor/cmake -DCMAKE_BUILD_TYPE=debug ./makecp sqladvisor /usr/local/binsqladvisor --helpUsage:  sqladvisor [OPTION...] sqladvisorSQL Advisor SummaryHelp Options:  -?, --help              Show help optionsApplication Options:  -f, --defaults-file     sqls file  -u, --username          username  -p, --password          password  -P, --port              port  -h, --host              host  -d, --dbname            database name  -q, --sqls              sqls  -v, --verbose           1:output logs 0:output nothing

3.2  导入测试数据

为了隐私考虑,线上表名屏蔽,以 tableA 和 tableB 代替。脱敏处理的表结构如下:

CREATE TABLE `tableA` ( \  `ID` int(10) unsigned NOT NULL AUTO_INCREMENT, \  `CATE` varchar(32) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `E_ID` char(18) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `RD` varchar(32) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `RU` varchar(32) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `MO` int(10) NOT NULL DEFAULT '0' COMMENT 'xxxx', \  `LID` varchar(32) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `LEVEL` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'xxxx', \  `GI` varchar(15) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `GT` datetime DEFAULT NULL COMMENT 'xxxx', \  `CL` varchar(32) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `ST` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'xxxx', \  `RES` varchar(64) NOT NULL DEFAULT '' COMMENT 'xxxx', \  PRIMARY KEY (`ID`), \  UNIQUE KEY `i_e_id` (`E_ID`), \  KEY `i_lid` (`LID`), \  KEY `i_rd_ru_mo` (`RD`,`RU`,`MO`) \) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='tableA'CREATE TABLE `tableB` ( \  `ID` int(11) NOT NULL AUTO_INCREMENT, \  `LID` varchar(32) NOT NULL COMMENT 'xxxx', \  `NAME` varchar(45) NOT NULL COMMENT 'xxxx', \  `CL` varchar(30) NOT NULL COMMENT 'xxxx', \  `TIME` datetime DEFAULT NULL COMMENT 'xxxx', \  `NUM` varchar(64) NOT NULL COMMENT 'xxxx', \  `SOUR` varchar(32) NOT NULL COMMENT 'xxxx', \  `GI` varchar(15) NOT NULL COMMENT 'xxxx', \  `TYPE` tinyint(4) NOT NULL COMMENT 'xxxx', \  `SID` int(11) NOT NULL COMMENT 'xxxx', \  `SEID` int(11) NOT NULL COMMENT 'xxxx', \  `NAME` varchar(20) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `ADD` varchar(255) NOT NULL COMMENT 'xxxx', \  `PO` varchar(11) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `QQ` varchar(20) NOT NULL COMMENT 'xxxx', \  `CATE` varchar(20) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `RU` varchar(32) NOT NULL DEFAULT '' COMMENT 'xxxx', \  `SC` int(10) unsigned NOT NULL DEFAULT '0' COMMENT 'xxxx', \  `LEVEL` varchar(32) DEFAULT '' COMMENT 'xxxx', \  PRIMARY KEY (`ID`), \  KEY `i_user` (`LID`,`CATE`), \  KEY `i_num` (`NUM`), \  KEY `i_cl` (`CL`) \) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='tableB'

导入数据量如下:

mysql> SELECT COUNT(*) FROM tableA;+----------+| COUNT(*) |+----------+|   122658 |+----------+1 row in set (0.06 sec)mysql> SELECT COUNT(*) FROM tableB;+----------+| COUNT(*) |+----------+|   979525 |+----------+1 row in set (0.23 sec)

3.3  执行测试

测试语句:

/usr/local/bin/sqladvisor -u root -p 'xxxx' -P 3306 -h xxx.xxx.xxx.xxx -d databaseA \-q "SELECT * FROM tableA WHERE LID = 'xxxx' ORDER BY GT DESC" -v 1

测试结果

2017-03-14 12:30:51 1923 [Note] 第1步: 对SQL解析优化之后得到的SQL:\select `*` AS `*` from `databaseA`.`tableA` where (`LID` = 'xxxx') \order by `GT` desc2017-03-14 12:30:51 1923 [Note] 第2步:开始解析where中的条件:(`LID` = 'xxxx')2017-03-14 12:30:51 1923 [Note] show index from tableA2017-03-14 12:30:51 1923 [Note] show table ST like 'tableA'2017-03-14 12:30:51 1923 [Note] select count(*) from ( select `LID` from `tableA` \    FORCE INDEX( i_E_ID ) order by E_ID DESC limit 10000) `tableA` \    where (`LID` = 'xxxx')2017-03-14 12:30:51 1923 [Note] 第3步:表tableA的行数:122879,limit行数:10000,\得到where条件中(`LID` = 'xxxx')的选择度:100002017-03-14 12:30:51 1923 [Note] 第4步:开始解析order by 条件2017-03-14 12:30:51 1923 [Note] 第5步:开始验证 字段GT是不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] show index from tableA where Key_name = 'PRIMARY' \and Column_name ='GT' and Seq_in_index = 12017-03-14 12:30:51 1923 [Note] 第6步:字段GT不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] 第7步:开始添加order by 字段2017-03-14 12:30:51 1923 [Note] 第8步:开始验证 字段GT是不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] show index from tableA where Key_name = 'PRIMARY' \and Column_name ='GT' and Seq_in_index = 12017-03-14 12:30:51 1923 [Note] 第9步:字段GT不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] 第10步:开始验证 字段LID是不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] show index from tableA where Key_name = 'PRIMARY' \and Column_name ='LID' and Seq_in_index = 12017-03-14 12:30:51 1923 [Note] 第11步:字段LID不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] 第12步:开始验证 字段LID是不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] show index from tableA where Key_name = 'PRIMARY' \and Column_name ='LID' and Seq_in_index = 12017-03-14 12:30:51 1923 [Note] 第13步:字段LID不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] 第14步:开始验证表中是否已存在相关索引。表名:tableA, \字段名:LID, 在索引中的位置:12017-03-14 12:30:51 1923 [Note] show index from tableA where Column_name ='LID' \and Seq_in_index =12017-03-14 12:30:51 1923 [Note] 第15步:开始验证 字段GT是不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] show index from tableA where Key_name = 'PRIMARY' \and Column_name ='GT' and Seq_in_index = 12017-03-14 12:30:51 1923 [Note] 第16步:字段GT不是主键。表名:tableA2017-03-14 12:30:51 1923 [Note] 第17步:开始验证表中是否已存在相关索引。\表名:tableA, 字段名:GT, 在索引中的位置:22017-03-14 12:30:51 1923 [Note] show index from tableA where \Column_name ='GT' and Seq_in_index =22017-03-14 12:30:52 1923 [Note] 第18步:开始输出表tableA索引优化建议:2017-03-14 12:30:52 1923 [Note] Create_Index_SQL:alter table tableA add index \idx_LID_GT(LID,GT)2017-03-14 12:30:52 1923 [Note] 第19步: SQLAdvisor结束!

 

四、结论

美团点评开源 SQL 优化工具 SQLAdvisor 得到的优化建议比较满意,建议在线上试用一段时间。这个工具的成本在于需要在线上 DB 安装相关依赖,如果确认采用,可以考虑在初始化 DB 服务器时部署此工具。

原文地址:https://dbarobin.com/2017/03/16/test-report-of-sqladvisor/

官方地址:https://github.com/Meituan-Dianping/SQLAdvisor

为了方便大家交流,本人开通了微信公众号和QQ群,QQ群:291519319,喜欢技术的一起来交流吧

转载于:https://my.oschina.net/u/3023401/blog/1793409

你可能感兴趣的文章
IKE IPSEC ×××
查看>>
Redis3.0 集群
查看>>
lvm 2(逻辑卷管理)基础详解
查看>>
Hyper-v Server在线调整虚拟硬盘大小
查看>>
jQuery UI 拖动(Draggable) - 延迟开始
查看>>
TokuDB引擎笔记
查看>>
mongodb 安装
查看>>
fedora上ARM-LINUX-GCC 编译器安装
查看>>
20160726
查看>>
BZOJ 1927 星际竞速
查看>>
数据库
查看>>
机器学习入门-决策树算法
查看>>
vue-router学习笔记(一)
查看>>
Linux的用户和用户组管理
查看>>
怎样让webservice在浏览器远程浏览时像在本地浏览一样有参数输入框
查看>>
COGS 2353 2355 2356 2358 有标号的DAG计数
查看>>
「WC2018」即时战略
查看>>
linux密码修改实验
查看>>
Xcode+OpenCV3.4.0 折腾(2)
查看>>
JAVA时间日期处理类,主要用来遍历两个日期之间的每一天。
查看>>