首页>会议文档 >

龙雪刚_SQL 优化利器之SQLAdvisor

page:
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor
龙雪刚_SQL 优化利器之SQLAdvisor

龙雪刚_SQL 优化利器之SQLAdvisor

所属会议:第七届数据技术嘉年华会议地点:北京


下载

手机看
活动家APP客户端

扫二维码下载
或点击下载
Android iOS

2128次
浏览次数
第七届数据技术嘉年华所有文档 袋鼠 镇伟_云上 Oracle 最佳实践 张辉_智能网络打造数据库平台高速路 RAC 曾令军_性能优化实战 邝志辉_系统级跟踪诊断方法及案例 李真旭_细致入微:Oracle 经典案例分享 罗海雄_一次性能优化隐含的秘密_罗海雄 侯圣文_大数据,未来职场必须要get的技能域 肖鹏_回首十年,如何让路变的更宽 - DBA 职场规划反思 杨建荣_Oracle 和 MySQL DBA 的进阶之路 云和开创 刘伟 -可读性的explain Andrew Holdsworth_Where is the industry going in the autonomous database era 王佳毅_永不停止的脚步 - 数据库优化之路 吴承杨_Autonomous DB and Oracle Paas IAAS strategy 楼方鑫_Oracle 数据库网关生产实践 彭晓平_大型数据中心数据安全管控 去哪儿 高级DBA 强昌金_靠谱才是硬道理 - MySQL 数据安全体系详解 黄廷忠_子查询优化的最佳实践 刘旭_SQL 审核服务 - 行业最佳实践 DSG创始人、总裁 韩宏坤_大数据时代的DSG复制云_部分1 DSG创始人、总裁 韩宏坤_大数据时代的DSG复制云_部分2 鲍琳_我们是如何构建金融级数据库云的 王向飞_京东云数据库架构实践 云和恩墨 杨廷琨_从自动化到自治数据库 - 从 Oracle 12c 到 18c 的新进展 郑战伟_云计算、大数据下灾备行业发展趋势探讨 周崇毅_Kubernetes 与 OpenStack 融合支撑企业级微服务架构 K-DB 江豫京_云技术发展历程及展望 TiDB 申砾_Design and Architecture 那海蓝蓝 李海翔_腾讯金融云分布式数据库云分布式数据库的事务处理技术 宋利兵_MySQL - 8-0 中的复制技术 张云杨_阿里云新一代数据库 POLARDB Oracle数据库团队 郑军-全面解读Oracle数据库云 福建农信 谢方恺_基于工具的 SQL 审核研究与实践 海通证券 王朝阳_海通证券金融云思考与实践 郝大为_银行 PB 级别海量非结构化数据管理实践 赵守斌_恒丰金融云探索实践 渤海财险数据架构智能化演进- 2017数据技术嘉年华脱敏 唐刘_分布式强一致性数据库的灵魂 - Raft 算法的理论和实践 王璞_下一代企业级应用架构管理体系 钟储建_后 IOE 时代的数据平台架构实践 程实_ElasticDB 数据库 偶数科技 常雷_云数据库的本质 杨琳_Oracle TimesTen下一代分布式内存数据库概述 高新刚_打造京东金融平稳高效的数据库运维体系 Andrew Holdsworth曲卓董志平_自治数据库时代的 SQL 优化

文档介绍

索引优化是常用的 SQL 优化方式之一,往往可以在短时间内产生非常巨大的效果。如果能够将索引优化工作转化成工具化、标准化的流程,减少人工介入,无疑会大大提高工作效率。如果结合线上实际情况,实时分析,绝对是线上性能保驾护航的利器。于是,SQLAdviosr 来了。

演讲实录

介绍

在数据库运维过程中,优化 SQL 是 DBA 团队的日常任务。例行 SQL 优化,不仅可以提升程序性能,还能够降低线上故障的概率。

目前常用的 SQL 优化方式包括但不限于:业务层优化、SQL逻辑优化、索引优化等。其中索引优化通常通过调整索引或新增索引从而达到 SQL 优化的目的。索引优化往往可以在短时间内产生非常巨大的效果。如果能够将索引优化转化成工具化、标准化的流程,减少人工介入的工作量,无疑会大大提高DBA的工作效率。

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

现在,我们非常高兴地将 SQLAdvisor 开源,项目 GitHub 地址:https://github.com/Meituan-Dianping/SQLAdvisor 。我们已经把相关开发工作全面转到 GitHub 上,开源版本和内部使用版本保持完全一致。希望与业内有类似需求的团队,一起打造一款优秀的 SQL 优化产品。

SQLAdvisor架构流程图:

SQLAdvisor使用举例

sql: SELECT id FROM crm_loan WHERE id_card = 1234567

cmd: ./sqladvisor -h xx -P xx -u xx -pxx -d xx -q "SELECT id FROM crm_loan WHERE id_card = 1234567 "

SQLAdvisor输出: alter table crm_loan add index idx_id_card(id_card)

SQLAdvisor的优点

基于 MySQL 原生词法解析,充分保证词法解析的性能、准确定以及稳定性;

支持常见的 SQL(Insert/Delete/Update/Select);

支持多表 Join 并自动逻辑选定驱动表;

支持聚合条件 Order by 和 Group by;

过滤表中已存在的索引。

SQLAdvisor原理介绍

Join 处理

Join语法分为两种:Join on 和 Join using,并且 Join on 有时会存在 where 条件中。

分析 Join 条件首先会得到一个 nested_join 的 table list,通过判断它的 join_using_fields 字段是否为空来区分 Join on 与 Join using。

生成的 table list 以二叉树的形式进行存储,以后序遍历的方式对二叉树进行遍历。

生成内部解析树时,right Join 会转换成 left Join。

Join 条件会存在当层的叶子节点上,如果左右节点都是叶子节点,会存在右叶子节点。

每一个非叶子节点代表一次 Join 的结果。

上述实现时,涉及的函数为:mysql_sql_parse_join(TABLE_LIST join_table) mysql_sql_parse_join(Item join_condition),主要流程图如下:

where 处理

主要是提取 SQL 语句的 where 条件。where 条件中一般由 AND 和 OR 连接符进行连接,因为 OR 比较难以处理,所以忽略,只处理 AND 连接符。

由于 where 条件中可以存在 Join 条件,因此需要进行区分。

依次获取 where 条件,当条件中的操作符是 like,如果不是前缀匹配则丢弃这个条件。

根据条件计算字段的区分度按照高低进行倒序排,如果小于30则丢弃。同时使用最左原则将 where 条件进行有序排列。

计算区分度

通过 “show table status like” 获得表的总行数 table_count。

通过计算选择表中已存在的区分度最高的索引 best_index,同时Primary key > Unique key > 一般索引。

通过计算获取数据采样的起始值offset与采样范围rand_rows:

offset = (table_count / 2) > 10W ? 10W : (table_count / 2)

rand_rows =(table_count / 2) > 1W ? 1W : (table_count / 2)

使用select count(1) from (select field from table force index(best_index) order by cl.. desc limit rand_rows) where field_print 得到满足条件的rows。

cardinality = rows == 0 ? rand_rows : rand_rows / rows;

计算完成选择度后,会根据选择度大小,将该条件添加到该表中的备选索引中。

主要涉及的函数为:mysql_sql_parse_field_cardinality_new()计算选择度。

添加备选索引

mysql_sql_parse_index()将条件按照选择度添加到备选索引链表中。

上述两函数的流程图如下所示:

Group 与 Order 处理

Group 字段与 Order 字段能否用上索引,需要满足如下条件:

涉及到的字段必须来自于同一张表,并且这张表必须是确定下来的驱动表。

Group by 优于 Order by, 两者只能同时存在一个。

Order by 字段的排序方向必须完全一致,否则丢弃整个 Order by 字段列。

当 Order by 条件中包含主键时,如果主键字段为 Order by。 字段列末尾,忽略该主键,否则丢弃整个 Order by 字段列。

整个索引列排序优先级:等值>(group by | order by )> 非等值。

该过程中设计的函数主要有:

mysql_sql_parse_group() 判断 Group 后的字段是否均来自于同一张表。

mysql_sql_parse_order() 判断 Order 后的条件是否可以使用。

mysql_sql_parse_group_order_add() 将字段依次按照规则添加到备选索引链表中。

驱动表选择

经过前期的 where 解析、Join 解析,已经将 SQL 中表关联关系存储起来,并且按照一定逻辑将候选驱动表确定下来。

在侯选驱动表中,按照每一张表的侯选索引字段中第一个字段进行计算表中结果集大小。

使用 explain select * from table where field 来计算表中结果集。

结果集小最小的被确为驱动表。

步骤中涉及的函数为:final_table_drived(),在该函数中,调用了函数 get_join_table_result_set() 来获取每张驱动候选表的行数。

添加被驱动表备选索引

通过上述过程,已经选择了驱动表,也通过解析保存了语句中的条件。

由于选定了驱动表,因此需要对被驱动表的索引,根据 Join 条件进行添加。

该过程涉及的函数主要是:mysql_index_add_condition_field(),流程如下:

输出建议

通过上述步骤,已经将每张表的备选索引键全部保存。此时,只要判断每张表中的候选索引键是否在实际表中已存在。没有索引,则给出建议增加对应的索引。

该步骤涉及的函数是:print_index() ,主要的流程图为:

SQLAdvisor版本更新

Functionality Added or Changed

调整架构将 SQLParser 与 SQLAdvisor 模块隔离,方便调试。

重新架构多表 Join 关系的 find_join_elements() 函数,思路更加清晰。

修改选定驱动表的策略,确保驱动表为小结果集。

添加 where 条件中的 like 处理。

优化 Order by 逻辑,忽略 Order by primary key 场景。

输出索引建议前,增加判断索引是否已存在。

Bugs Fixed

修复 SQL 无法处理中文问题。

修复字段多次出现在 where 条件中从而导致多次出现在索引列中问题。

修复在 find_best_index() 函数中,对 MySQL API 中的 result 对象提前 free,导致指针失效问题。

×

打开微信扫一扫,分享到朋友圈