浅谈数据库优化

浅谈数据库优化

在面试的时候经常会被问到数据库如何优化的?面试的时候最担心遇到这种宽泛的问题。如果你直接说:“建索引” 类似这样简单的回答,个人感觉这种回答很不专业,面试官也不会喜欢这样的回答。虽然我们不需要像DBA一样对数据库有非常深入的研究,但是至少应该对这种问题有个系统全面的回答会比较好。
当然,我们不单单是为了面试,在实际生产中我们也需要知道一些最基本的优化方法。就在最近我们组依赖的服务线上就突然出现挂掉的情况,很久才恢复,后来发现是出现慢查询导致数据库连接池撑爆了,所以掌握数据库优化技巧是非常重要的。在这里就总结几条优化方法。
PS:下面介绍的方法都是针对MySQL数据库优化。

MySQL 架构解析

MySQL架构解析个人感觉这篇文章分析的非常透彻详细。这里就不照搬过来了。

优化策略

数据类型优化

数据类型的优化主要是指选取什么类型。需要遵循“小而简单”的原则。因为这样的数据类型占用的内存、磁盘更低,CPU处理时间也更少。举个常见的例子。

1、日期类型选择。MySQL中关于时间类型,MySQL中最小精度是秒。有DatetimeTimestampint三种类型类型来存储时间,个人推荐使用DatetimeDatetime时间类型存储的范围比Timestamp更大,而且Timestamp因时区不同而不同,int时间显示不够直观。

2、指定最大显示宽度,不会改变存储空间。显示宽度与存储大小或类型包含的值的范围无关。也就是说对存储和计算来讲,指定了长度的int(1)int(12)是相同的。

3、 unsigned(无符号)属性不允许有负值,这可以使正数存储范围扩大一倍,比如UNSIGNED TINYINT存储的范围是0 - 255,TINYINT的范围是-128 ~ 127。所以在没有用到负数的情况下,建议用unsigned(无符号)

4、存储IP地址时最好使用无符号整数,而不是字符串,这样可以节省存储空间,Inet_ATON()将带点儿的IP转为数字,而Inet_NTOA可将数字转为IP。

5、对一些精度要求比较高的数据,有人建议使用DECIMAL,decimal需要额外的空间和计算开销。建议使用BIGINT,在需要精确到千分之一的时候,可以先乘以1000,再用BIGINT存。

6、通常情况下列最好为NOT NULL, NULL 会使得索引失效。

7、当数据量比较大的时候,不推荐使用alter table。因为alter table 会创建一个新结构的表,并把老表中的数据插入到新表中。

8、不推荐使用Enum。因为枚举类型是固定的字符串列表,添加和删除的时候需要使用alter table命令。

索引优化

MySQL优化最重要建立索引,建立好的索引可以起到事半功倍的效果。

索引结构与算法

首先要介绍索引的结构。索引数据结构与原理可以参考这篇文章:MySQL索引背后的数据结构及算法原理。这里不做过多的解释。
看过这篇文章之后抛出一个问题:为什么不用Hash索引,而是使用B-Tree索引结构,理论上Hash索引的查询时间复杂度只有O(1)。这里主要有以下5点原因:
1、hash函数计算后的结果,是随机的,如果是在磁盘上放置数据。比主键为id为例, 那么随着id的增长, id对应的行,在磁盘上随机放置.
2、不法对范围查询进行优化。
3、无法利用前缀索引. 比如 在btree中, field列的值“hellopworld”,并加索引。查询 xx=helloword,自然可以利用索引, xx=hello,也可以利用索引,(左前缀索引)。因为hash(‘helloword’),和hash(‘hello’),两者的关系仍为随机。
4、排序也无法优化。
5、必须回行.就是说 通过索引拿到的只是数据位置,而不是数据值,必须回到表中取数据。

索引注意点

  • 理想的索引应该具有下面几点:
    1: 查询频繁 2: 区分度高 3: 长度小 4: 尽量能覆盖常用查询字段。

  • 哪些列不应该建立索引呢?
    1、更新非常频繁的字段不适合建立索引。2、唯一性太差的字段不适合单独创建索引。3、不会出现在where子句中的字段不应该创建索引。

  • 索引覆盖
    索引覆盖是指 如果查询的列恰好是索引的一部分,那么查询只需要在索引文件上进行,不需要回行到磁盘再找数据.
    这种查询速度非常快,称为”索引覆盖”

  • 不要用UUID或者随机字符串作为主键值,尽量用连续增长的值
    对于innodb而言,因为节点下有数据文件,因此节点的分裂将会比较慢。对于innodb的主键,尽量用整型,而且是递增的整型。如果是无规律的数据,将会产生页的分裂,影响速度。关于UUID与自增主键的比较参看这篇文章

  • 索引不是创建的越多越好。过多的索引不可能一次性读取到内存,索引是以索引文件的形式存储在磁盘上。过多的索引是会产生磁盘I/O消耗,从而影响性能。

特定语句优化

Count化

MyIsam的count()比较快,原因是MyIsam对行数进行了存储。一旦有条件的查询, 速度就不再快了,尤其是where条件的列上没有索引。
假如,id<100的商家都是我们内部测试的,我们想查查真实的商家有多少?
select count(*) from lx_com where id>=100; (1000多万行用了6.X秒)
小技巧:

select count(*) from lx_com; 快
select count(*) from lx_com where id<100; 快

Join优化

Join语句经常听到“小结果集驱动大结果集”。为什么会有这句话?这和Mysql的关联查询原理有关。 Mysql的关联查询是取第一张表的一行数据去遍历第二张表的所有数据找到匹配的行,依次遍历第一张表的数据。有人会问MN和NM结果不是一样的吗?“小结果集驱动大结果集”,这句话的前提是连接字段建立了索引。具体可以看下面的例子:

SELECT A.xx,B.yy 
FROM A INNER JOIN B USING(c)
WHERE A.xx IN (5,6)

上述代码的执行可以参照下面的伪代码:

outer_iterator = SELECT A.xx,A.c FROM A WHERE A.xx IN (5,6);
outer_row = outer_iterator.next;
while(outer_row) {
    inner_iterator = SELECT B.yy FROM B WHERE B.c = outer_row.c;
    inner_row = inner_iterator.next;
    while(inner_row) {
        output[inner_row.yy,outer_row.xx];
        inner_row = inner_iterator.next;
    }
    outer_row = outer_iterator.next;
}

假设我们在A和B表的c列都建立了索引,可以看到外层循环无法用到索引一定会遍历M次,但是内层循环可以利用索引减少内存循环的次数,如果B的数据量大的话,优化效果还是非常可观的。

group by优化

Group by的实质是先排序然后分组。所以建议在group by中利用索引,这样可以减少临时表的创建以及文件排序。order by的列要和group by的一致,否则也会引起临时表
(原因是因为group by 和 order by 都需要排序,如果2者的列不一致,那必须经过至少1次排序)。以A,B表连接为例 ,主要查询A表的列, 那么 group by ,order by 的列尽量相同,而且列应该显示声明为A的列

select A.id,A.cat_id from A inne join B group by A.cat_id order by A.cat_id

Limit优化

当表的数据非常多的时候,limit的分页优化可以用延迟索引。比如我们要查询5000000后的10条记录,用下面的语句效率是非常低的。

select id,name from lx_com limit 5000000,10;

这是因为limit offset,N, 当offset非常大时, 效率极低,
可以先在子查询语句里利用覆盖索引扫描,然后再做一个关联查询,这种技术就是延迟索引
。SQL语句如下:

select id,name from lx_com inner join (select id from lx_com limit 5000000,10) as tmp using(id);

Union优化

建议使用union all 而不是union。union all 不过滤 效率提高,如非必须,请用union all。因为 union去重的代价非常高, Mysql会把各个查询结果插入到临时表中,然后做唯一性检查。所以请放在程序里去重。

性能分析工具

开启慢查询日志

image.png

MySQL 慢查询的相关参数解释:

  • slow_query_log :是否开启慢查询日志,1表示开启,0表示关闭。

  • log-slow-queries :旧版(5.6以下版本)MySQL数据库慢查询日志存储路径。可以不设置该参数,系统则会默认给一个缺省的文件host_name-slow.log

  • slow-query-log-file:新版(5.6及以上版本)MySQL数据库慢查询日志存储路径。可以不设置该参数,系统则会默认给一个缺省的文件host_name-slow.log

  • long_query_time :慢查询阈值,当查询时间多于设定的阈值时,记录日志。

使用profile语句

打开profile分析语句


image.png

查看分析列表


image.png

查看单条语句的执行过程


image.png

使用explain语句

如果要定量分析查询语句涉及到了多少行,可以使用explainexplain语句是非常重要的分析工具。提测之前使用explain分析一下SQL语句是一种美德。explain可以显示如下字段:

image.png

关于每个字段的值的解释可以参考这篇文章

这里会选择我认为比较重要几个字段的值通过列子进行讲解。

select_type 查询类型

  • simple语句中没有子查询或者union
select field from table;
  • dependent subquery 子查询中的第一个select语句,依赖外部查询结果集
select * from test.tabname where id in(select id from test.tabname2 where name='love');

以上语句有个错误的理解是认为按照下面两个结果执行

select group_concat(id) from test.tabname2 where name='love';
--内层查询结果:1,3,5,7,9,11,13,15,17,1
 
select * from test.tabname where id in(1,3,5,7,9,11,13,15,17,19);
image.png

通过explain发现其实他是先根据关联外部tabname, 而不是先去执行这个子查询。可以通过连接查询来优化上面的语句。

select tabname. * from test.tabname inner join test.tabname2 using(id) where tabname2.name='love';
image.png

可以看到查询类型变成了simple简单查询。

  • primary最外层的select, 例子参见dependent subquery

type:很重要,显示了连接使用了哪种类别,有无使用索引。type代表查询执行计划(QEP)中指定的表使用的连接方式

从最好到最差依次为::system,const,eq_ref,ref,fulltext,ref_or_null,unique_subquery,index_subquery,range,index_merge,index,ALL

  • system: const的一个特例,表中只有一条记录

  • const: where条件是以常量为单位,表中最多一条记录匹配。


    image.png
  • eq_ref:最多只会有一条匹配结果,一般是通过主键或是唯一索引来访问。一般会出现在连接查询的语句中。通过索引列,直接引用某1行数据

    image.png

  • ref: 它返回所有匹配某个单个值的行。它用在关联操作只使用了索引的最左前缀,或者索引不是UNIQUE和PRIMARY KEY。连接查询中被驱动的表索引引用查询.。通过索引列,可以直接引用到某些数据行

    image.png

  • range: 引用范围扫描,见上面的例子

  • all: 全表扫描效率最低

possible key 可能用到哪些索引进行查询

key 实际用到的索引

key_len 实际用到的索引字节数

关于key_len大小的计算可以参照这篇文章

ref 列出是通过哪个字段来进行连接查询,或者是否是通过常量(const)

extra 是EXPLAIN输出中另外一个很重要的列,该列显示MySQL在查询过程中的一些详细信息。

重点关注下面几个值:

  • Using filesort:文件排序(文件可能在磁盘,也可能在内存)。

  • Using temporary:是指用上了临时表, group by 与order by 不同列时,或group by ,order by 别的表的列.

  • Using index:表示使用索引,如果只有 Using index,说明他没有查询到数据表,只用索引表就完成了这个查询,这个叫覆盖索引,效率非常高。

  • Using index condition:意味着查询列的某一部分无法直接使用索引

  • Using where:过滤元素的时候出现,也会扫描表,但是如果在条件语句中存在索引列,会优先使用带索引的条件。explain SELECT * from test where b = '4' (b不是索引,全表扫描后,通过过滤获取所需数据)

总结

本文先介绍了MySQL的架构,然后从数据类型、索引、性能分析三个角度描述如何进行数据库优化。相信当面试官再问你如何进行数据库优化的时候。你不会简单的回答“建索引”。最后提醒大家:提测前请explain

数据库优化是常见的面试题,更多资料请见:https://zfau4tzxp5.feishu.cn/docx/ByM0dxtWoo87QtxnWtTcG0kCnpg
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容