CategoryResourceRepost/极客时间专栏/SQL必知必会/第二章:SQL性能优化篇/20丨当我们思考数据库调优的时候,都有哪些维度可以选择?.md
louzefeng d3828a7aee mod
2024-07-11 05:50:32 +00:00

174 lines
16 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<audio id="audio" title="20丨当我们思考数据库调优的时候都有哪些维度可以选择" controls="" preload="none"><source id="mp3" src="https://static001.geekbang.org/resource/audio/55/ec/55d54688c3f64ce3ad39fdd774df65ec.mp3"></audio>
从这一篇开始我们正式进入了SQL性能优化篇。在这一模块中我们会关注如何提升SQL查询的效率。你可以思考一下如何你是一名DBA或者开发人员都有哪些维度可以进行数据库调优
其实关于数据库调优的知识点非常分散。不同的DBMS不同的公司不同的职位不同的项目遇到的问题都不尽相同。为了能让你对数据库调优有一个整体的概览我把这些知识点做了一个梳理希望能对你有一些帮助。
今天的课程你需要掌握以下几个方面的内容:
1. 数据库调优的目标是什么?
1. 如果要进行调优,都有哪些维度可以选择?
1. 如何思考和分析数据库调优这件事?
## 数据库调优的目标
简单来说,数据库调优的目的就是要让数据库运行得更快,也就是说响应的时间更快,吞吐量更大。
不过随着用户量的不断增加以及应用程序复杂度的提升我们很难用“更快”去定义数据库调优的目标因为用户在不同时间段访问服务器遇到的瓶颈不同比如双十一促销的时候会带来大规模的并发访问还有用户在进行不同业务操作的时候数据库的事务处理和SQL查询都会有所不同。因此我们还需要更加精细的定位去确定调优的目标。
如何确定呢?一般情况下,有两种方式可以得到反馈。
### 用户的反馈
用户是我们的服务对象,因此他们的反馈是最直接的。虽然他们不会直接提出技术建议,但是有些问题往往是用户第一时间发现的。我们要重视用户的反馈,找到和数据相关的问题。
### 日志分析
我们可以通过查看数据库日志和操作系统日志等方式找出异常情况,通过它们来定位遇到的问题。
除了这些具体的反馈以外,我们还可以通过监控运行状态来整体了解服务器和数据库的运行情况。
### 服务器资源使用监控
通过监控服务器的CPU、内存、I/O等使用情况可以实时了解服务器的性能使用与历史情况进行对比。
### 数据库内部状况监控
在数据库的监控中活动会话Active Session监控是一个重要的指标。通过它你可以清楚地了解数据库当前是否处于非常繁忙的状态是否存在SQL堆积等。
除了活动会话监控以外,我们也可以对事务、锁等待等进行监控,这些都可以帮助我们对数据库的运行状态有更全面的认识。
## 对数据库进行调优,都有哪些维度可以进行选择?
我们需要调优的对象是整个数据库管理系统它不仅包括SQL查询还包括数据库的部署配置、架构等。从这个角度来说我们思考的维度就不仅仅局限在SQL优化上了。
听起来比较复杂,但其实我们可以一步步通过下面的步骤进行梳理。
### 第一步选择适合的DBMS
我们之前讲到了SQL阵营和NoSQL阵营。在RDBMS中常用的有OracleSQL Server和MySQL等。如果对事务性处理以及安全性要求高的话可以选择商业的数据库产品。这些数据库在事务处理和查询性能上都比较强比如采用SQL Server那么单表存储上亿条数据是没有问题的。如果数据表设计得好即使不采用分库分表的方式查询效率也不差。
除此以外你也可以采用开源的MySQL进行存储我们之前讲到过它有很多存储引擎可以选择如果进行事务处理的话可以选择InnoDB非事务处理可以选择MyISAM。
NoSQL阵营包括键值型数据库、文档型数据库、搜索引擎、列式存储和图形数据库。这些数据库的优缺点和使用场景各有不同比如列式存储数据库可以大幅度降低系统的I/O适合于分布式文件系统和OLAP但如果数据需要频繁地增删改那么列式存储就不太适用了。原因我在答疑篇已经讲过这里不再赘述。
DBMS的选择关系到了后面的整个设计过程所以第一步就是要选择适合的DBMS。如果已经确定好了DBMS那么这步可以跳过但有时候我们要根据业务需求来进行选择。
### 第二步,优化表设计
选择了DBMS之后我们就需要进行表设计了。RDBMS中每个对象都可以定义为一张表表与表之间的关系代表了对象之间的关系。如果用的是MySQL我们还可以根据不同表的使用需求选择不同的存储引擎。除此以外还有一些优化的原则可以参考
1. 表结构要尽量遵循第三范式的原则(关于第三范式,我在后面章节会讲)。这样可以让数据结构更加清晰规范,减少冗余字段,同时也减少了在更新,插入和删除数据时等异常情况的发生。
1. 如果分析查询应用比较多,尤其是需要进行多表联查的时候,可以采用反范式进行优化。反范式采用空间换时间的方式,通过增加冗余字段提高查询的效率。
1. 表字段的数据类型选择关系到了查询效率的高低以及存储空间的大小。一般来说如果字段可以采用数值类型就不要采用字符类型字符长度要尽可能设计得短一些。针对字符类型来说当确定字符长度固定时就可以采用CHAR类型当长度不固定时通常采用VARCHAR类型。
数据表的结构设计很基础,也很关键。好的表结构可以在业务发展和用户量增加的情况下依然发挥作用,不好的表结构设计会让数据表变得非常臃肿,查询效率也会降低。
### 第三步,优化逻辑查询
当我们建立好数据表之后,就可以对数据表进行增删改查的操作了。这时我们首先需要考虑的是逻辑查询优化,什么是逻辑查询优化呢?
SQL查询优化可以分为逻辑查询优化和物理查询优化。逻辑查询优化就是通过改变SQL语句的内容让SQL执行效率更高效采用的方式是对SQL语句进行等价变换对查询进行重写。重写查询的数学基础就是关系代数。
SQL的查询重写包括了子查询优化、等价谓词重写、视图重写、条件简化、连接消除和嵌套连接消除等。
比如我们在讲解EXISTS子查询和IN子查询的时候会根据小表驱动大表的原则选择适合的子查询。在WHERE子句中会尽量避免对字段进行函数运算它们会让字段的索引失效。
我举一个例子假设我想对商品评论表中的评论内容进行检索查询评论内容开头为abc的内容都有哪些如果在WHERE子句中使用了函数语句就会写成下面这样
```
SELECT comment_id, comment_text, comment_time FROM product_comment WHERE SUBSTRING(comment_text, 1,3)='abc'
```
我们可以采用查询重写的方式进行等价替换:
```
SELECT comment_id, comment_text, comment_time FROM product_comment WHERE comment_text LIKE 'abc%'
```
你会发现在数据量大的情况下第二条SQL语句的查询效率要比前面的高很多执行时间为前者的1/10。
### 第四步,优化物理查询
物理查询优化是将逻辑查询的内容变成可以被执行的物理操作符,从而为后续执行器的执行提供准备。它的核心是高效地建立索引,并通过这些索引来做各种优化。
但你要知道索引不是万能的,我们需要根据实际情况来创建索引。那么都有哪些情况需要考虑呢?
1. 如果数据重复度高就不需要创建索引。通常在重复度超过10%的情况下,可以不创建这个字段的索引。比如性别这个字段(取值为男和女)。
1. 要注意索引列的位置对索引使用的影响。比如我们在WHERE子句中对索引字段进行了表达式的计算会造成这个字段的索引失效。
1. 要注意联合索引对索引使用的影响。我们在创建联合索引的时候会对多个字段创建索引这时索引的顺序就很重要了。比如我们对字段x, y, z创建了索引那么顺序是(x,y,z)还是(z,y,x),在执行的时候就会存在差别。
1. 要注意多个索引对索引使用的影响。索引不是越多越好,因为每个索引都需要存储空间,索引多也就意味着需要更多的存储空间。此外,过多的索引也会导致优化器在进行评估的时候增加了筛选出索引的计算时间,影响评估的效率。
查询优化器在对SQL语句进行等价变换之后还需要根据数据表的索引情况和数据情况确定访问路径这就决定了执行SQL时所需要消耗的资源。SQL查询时需要对不同的数据表进行查询因此在物理查询优化阶段也需要确定这些查询所采用的路径具体的情况包括
1. 单表扫描:对于单表扫描来说,我们可以全表扫描所有的数据,也可以局部扫描。
1. 两张表的连接常用的连接方式包括了嵌套循环连接、HASH连接和合并连接。
1. 多张表的连接:多张数据表进行连接的时候,顺序很重要,因为不同的连接路径查询的效率不同,搜索空间也会不同。我们在进行多表连接的时候,搜索空间可能会达到很高的数据量级,巨大的搜索空间显然会占用更多的资源,因此我们需要通过调整连接顺序,将搜索空间调整在一个可接收的范围内。
物理查询优化是在确定了逻辑查询优化之后,采用物理优化技术(比如索引等),通过计算代价模型对各种可能的访问路径进行估算,从而找到执行方式中代价最小的作为执行计划。在这个部分中,我们需要掌握的重点是对索引的创建和使用。
### 第五步使用Redis或Memcached作为缓存
除了可以对SQL本身进行优化以外我们还可以请外援提升查询的效率。
因为数据都是存放到数据库中,我们需要从数据库层中取出数据放到内存中进行业务逻辑的操作,当用户量增大的时候,如果频繁地进行数据查询,会消耗数据库的很多资源。如果我们将常用的数据直接放到内存中,就会大幅提升查询的效率。
键值存储数据库可以帮我们解决这个问题。
常用的键值存储数据库有Redis和Memcached它们都可以将数据存放到内存中。
从可靠性来说Redis支持持久化可以让我们的数据保存在硬盘上不过这样一来性能消耗也会比较大。而Memcached仅仅是内存存储不支持持久化。
从支持的数据类型来说Redis比Memcached要多它不仅支持key-value类型的数据还支持ListSetHash等数据结构。 当我们有持久化需求或者是更高级的数据处理需求的时候就可以使用Redis。如果是简单的key-value存储则可以使用Memcached。
通常我们对于查询响应要求高的场景响应时间短吞吐量大可以考虑内存数据库毕竟术业有专攻。传统的RDBMS都是将数据存储在硬盘上而内存数据库则存放在内存中查询起来要快得多。不过使用不同的工具也增加了开发人员的使用成本。
### 第六步,库级优化
库级优化是站在数据库的维度上进行的优化策略,比如控制一个库中的数据表数量。另外我们可以采用主从架构优化我们的读写策略。
如果读和写的业务量都很大并且它们都在同一个数据库服务器中进行操作那么数据库的性能就会出现瓶颈这时为了提升系统的性能优化用户体验我们可以采用读写分离的方式降低主数据库的负载比如用主数据库master完成写操作用从数据库slave完成读操作。
除此以外我们还可以对数据库分库分表。当数据量级达到亿级以上时有时候我们需要把一个数据库切成多份放到不同的数据库服务器上减少对单一数据库服务器的访问压力。如果你使用的是MySQL就可以使用MySQL自带的分区表功能当然你也可以考虑自己做垂直切分和水平切分。
什么情况下做垂直切分,什么情况下做水平切分呢?
如果数据库中的数据表过多,可以采用垂直分库的方式,将关联的数据表部署在一个数据库上。
如果数据表中的列过多,可以采用垂直分表的方式,将数据表分拆成多张,把经常一起使用的列放到同一张表里。
如果数据表中的数据达到了亿级以上可以考虑水平切分将大的数据表分拆成不同的子表每张表保持相同的表结构。比如你可以按照年份来划分把不同年份的数据放到不同的数据表中。2017年、2018年和2019年的数据就可以分别放到三张数据表中。
采用垂直分表的形式,就是将一张数据表分拆成多张表,采用水平拆分的方式,就是将单张数据量大的表按照某个属性维度分成不同的小表。
但需要注意的是,分拆在提升数据库性能的同时,也会增加维护和使用成本。
## 我们该如何思考和分析数据库调优这件事
做任何事情之前,我们都需要确认目标。在数据库调优中,我们的目标就是响应时间更快,吞吐量更大。利用宏观的监控工具和微观的日志分析可以帮我们快速找到调优的思路和方式。
虽然每个人的情况都不一样,但我们同样需要对数据库调优这件事有一个整体的认知。在思考数据库调优的时候,可以从三个维度进行考虑。
**首先,选择比努力更重要。**
在进行SQL调优之前可以先选择DBMS和数据表的设计方式。你能看到不同的DBMS直接决定了后面的操作方式数据表的设计方式也直接影响了后续的SQL查询语句。
**另外你可以把SQL查询优化分成两个部分逻辑查询优化和物理查询优化。**
虽然SQL查询优化的技术有很多但是大方向上完全可以分成逻辑查询优化和物理查询优化两大块。逻辑查询优化就是通过SQL等价变换提升查询效率直白一点就是说换一种查询写法执行效率可能更高。物理查询优化则是通过索引和表连接方式等技术来进行优化这里重点需要掌握索引的使用。
**最后,我们可以通过外援来增强数据库的性能。**
单一的数据库总会遇到各种限制,不如取长补短,利用外援的方式。
另外通过对数据库进行垂直或者水平切分,突破单一数据库或数据表的访问限制,提升查询的性能。
本篇文章中涉及到的概念和知识点比较多,也有可能出现纰漏,不过没有关系,我会在在后续的文章中陆续进行讲解。希望这篇文章可以让你站在一个宏观的角度对数据库的调优有系统性的认知,对今后的工作有一些启发。
<img src="https://static001.geekbang.org/resource/image/d3/b0/d3bc10314c3532f053304a00765183b0.jpg" alt=""><br>
你不妨说一下在日常的工作中你是如何发现数据库性能瓶颈的又是怎么解决这个问题的另外我在文章中从6个维度阐述了如何对数据库进行调优前两个维度在于选择中间两个维度在于SQL的查询优化后两个维度在于外援技术。你可以说一说你对这些维度的理解吗
欢迎你在评论区分享你的心得,也欢迎把这篇文章分享给你的朋友或者同事。