MySQL查询性能优化总结

MySQL查询性能优化总结

查询执行路径

  1. 客户端发送一条查询给MySQL服务器
  2. 服务器先检查缓存,如果命中了缓存,则立刻返回缓存中的结果,否则进行下一阶段
  3. 服务器端进行SQL解析、预处理,再由优化器生成对应的执行计划
  4. MySQL根据优化器生成的执行计划,调用存储引擎的API来执行查询。
  5. 将结果返回给客户端

优化数据访问

不要向数据库请求不需要的数据,例如多余的数据行、多余的字段、多表关联返回所有列、多次取重复数据。

MySQL使用如下三种方式应用WHERE条件,从好到坏以此为:

  • 在索引中使用WHERE条件过滤不匹配的记录。这是在存储引擎层完成的。
  • 使用索引覆盖扫描(在Extra列中出现了Using index)来返回记录,直接从索引中过滤不需要的记录并返回命中的结果,这是在MySQL服务层完成的,但无须再回表查询记录。
  • 从数据表中返回数据,然后过滤不满足条件(在Extra列中出现Using Where)。这在MySQL服务层完成,MySQL需要先从数据表读出记录然后过滤。

重构查询方式

  • 将复杂查询拆分成多个不同功能的简单查询。现在不用受限于网络通信、带宽等因素,且MySQL的连接和断开都是轻量级的,所以运行多个小查询已经很容易,但具体业务还是要多实践。

  • 切分查询。将一个大的查询切分成每个查询功能都一样的小查询。例如定期清楚大量数据时,如果用大查询则可能需要一次锁住很多数据、占满整个事物日志、耗尽系统资源、阻塞很多小但重要的查询,这时我们则可以切分下大SQL。

    -- 一次执行大的查询
    DELETE FROM notes WHERE createdAt < DATE_SUB(NOW(), INTERVAL 3 MONTH)
    -- 多次执行同样功能的小查询,可以使用存储过程写循环,也可以在业务中做循环,下面示例是在业务中循环的
    -- 这样分批次删除,则可以减轻服务器的压力
    const rows_affected = 0
    do{
    rows_affected = do_query("DELETE FROM notes WHERE createdAt < DATE_SUB(NOW(), INTERVAL 3 MONTH) LIMIT 10000")
    } WHERE rows_affected > 0
    
  • 分解关联查询。很多对性能要求较高的应用都会对关联应用进行分解。

    SELECT * FROM tag
      JOIN tag_post ON tag_post.tag_id=tag.id
      JOIN post ON tag_post.post_id=post.id
    WHERE tag.tag='mysql';
    

    可以分解成下面的SQL。

    SELECT * FROM tag WHERE tag='mysql';
    SELECT * FROM tag_post WHERE tag_id=1234;
    SELECT * FROM post WHERE post.id IN (123,456,234,789);
    

    分解的好处:

    • 让缓存的效率更高,许多应用程序可以缓存单表查询结果对象
    • 执行单个查询,可以减少锁的竞争
    • 在应用层关联,可以更容易对数据库进行拆分,更容易做到高性能和可扩展
    • 查询本身效率也可能会提升
    • 可以减少冗余记录的查询

查询执行基础

  1. MySQL客户端/服务器通信协议

    客户端与服务器是“半双工”形式,在任意时刻,要么由S向C发送数据,要么由C向S发送数据,这两个动作不能同时发生。这也导致当C向S发送超长查询语句时,需要关注max_allowed_packet参数,S向C响应大量数据时,不仅数据库服务器需要占用大量时间计算和大量内存来保存结果,一次性返回给系统服务器,还会占用系统服务器的大量内存,而且C必须接收完,才能再释放这条查询所占用的资源,所以通常的做法是再应用系统中通过流查询,一部分一部分的接收数据。

查询优化

  1. 关联子查询,WHERE后IN和EXISTS的选择,NOT IN和 NOT EXISTS的选择

    外表数据集大,内表查询数据集小,使用IN,因为MySQL5.7默认200条数据内,IN会使用索引

    外表数据集小,内表查询数据集大,使用EXISTS,MySQL的嵌套循环优化优势更大

    NOT IN不会使用索引,NOT EXISTS子查询会使用到索引,无论外表大还是内表大NOT EXISTS效率都比NOT IN高。但世事无绝对,且关乎MySQL版本问题,遇到和类问题,还是需要多手动测试。

  2. UNION的限制

    如果UNION的各个子句能够根据LIMIT只取部分结果集,或者希望先排好序再合并结果集。我们可以先排好各个子句的顺序并且取限制条数,而不用先合并多个子句,再排序取LIMIT条数。这样可以避免UNION生成一个很大的数据集中间表。

    -- 如果actor表有1000条数据,customer有1000条数据,则会生成2000条临时表的数据,但我们却只需要20条数据
    (
    SELECT first_name, last_name
    FROM actor 
    ORDER BY last_name
    )
    UNION ALL
    (
    SELECT first_name, last_name
    FROM customer 
    ORDER BY last_name
    )
    LIMIT 20
    

    减少临时表数据的SQL

    -- 这样临时表就只用存储40条数据了
    (
    SELECT first_name, last_name
    FROM actor 
    ORDER BY last_name
    LIMIT 20
    )
    UNION ALL
    (
    SELECT first_name, last_name
    FROM customer 
    ORDER BY last_name
    LIMIT 20
    )
    LIMIT 20
    

上面只是知识点的梳理,后期项目中遇到典型的优化案例,我会持续更新进来。

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

推荐阅读更多精彩内容

  • 国庆第二天,今天嗨翻模式的开启就等室友的同学来了再说吧,在这之前,先来一波笔记 查询真正重要的是响应时间,查询包含...
    小炼君阅读 1,701评论 0 50
  • 1. 了解SQL 1.1 数据库基础 ​ 学习到目前这个阶段,我们就需要以某种方式与数据库打交道。在深入学习MyS...
    锋享前端阅读 1,062评论 0 1
  • 一. Java基础部分.................................................
    wy_sure阅读 3,810评论 0 11
  • 网上关于SQL优化的教程很多,都是从理论或者实际操作经验直接入手,今天我将已实际项目过程为主线来谈一谈MySql的...
    YingxiangEmpire阅读 475评论 0 2
  • 富书写作情报 每日分享新媒体写作圈最新资讯 2018年02月06日 周二 【晨语:真正的读书使瞌睡者醒来,给未定目...
    富书号阅读 206评论 0 0