MySQL实战45讲
林晓斌
网名丁奇,前阿里资深技术专家
立即订阅
43178 人已学习
课程目录
已完结 48 讲
0/4登录后,你可以任选4讲全文学习。
开篇词 (1讲)
开篇词 | 这一次,让我们一起来搞懂MySQL
免费
基础篇 (8讲)
01 | 基础架构:一条SQL查询语句是如何执行的?
02 | 日志系统:一条SQL更新语句是如何执行的?
03 | 事务隔离:为什么你改了我还看不见?
04 | 深入浅出索引(上)
05 | 深入浅出索引(下)
06 | 全局锁和表锁 :给表加个字段怎么有这么多阻碍?
07 | 行锁功过:怎么减少行锁对性能的影响?
08 | 事务到底是隔离的还是不隔离的?
实践篇 (37讲)
09 | 普通索引和唯一索引,应该怎么选择?
10 | MySQL为什么有时候会选错索引?
11 | 怎么给字符串字段加索引?
12 | 为什么我的MySQL会“抖”一下?
13 | 为什么表数据删掉一半,表文件大小不变?
14 | count(*)这么慢,我该怎么办?
15 | 答疑文章(一):日志和索引相关问题
16 | “order by”是怎么工作的?
17 | 如何正确地显示随机消息?
18 | 为什么这些SQL语句逻辑相同,性能却差异巨大?
19 | 为什么我只查一行的语句,也执行这么慢?
20 | 幻读是什么,幻读有什么问题?
21 | 为什么我只改一行的语句,锁这么多?
22 | MySQL有哪些“饮鸩止渴”提高性能的方法?
23 | MySQL是怎么保证数据不丢的?
24 | MySQL是怎么保证主备一致的?
25 | MySQL是怎么保证高可用的?
26 | 备库为什么会延迟好几个小时?
27 | 主库出问题了,从库怎么办?
28 | 读写分离有哪些坑?
29 | 如何判断一个数据库是不是出问题了?
30 | 答疑文章(二):用动态的观点看加锁
31 | 误删数据后除了跑路,还能怎么办?
32 | 为什么还有kill不掉的语句?
33 | 我查这么多数据,会不会把数据库内存打爆?
34 | 到底可不可以使用join?
35 | join语句怎么优化?
36 | 为什么临时表可以重名?
37 | 什么时候会使用内部临时表?
38 | 都说InnoDB好,那还要不要使用Memory引擎?
39 | 自增主键为什么不是连续的?
40 | insert语句的锁为什么这么多?
41 | 怎么最快地复制一张表?
42 | grant之后要跟着flush privileges吗?
43 | 要不要使用分区表?
44 | 答疑文章(三):说一说这些好问题
45 | 自增id用完怎么办?
特别放送 (1讲)
直播回顾 | 林晓斌:我的 MySQL 心路历程
结束语 (1讲)
结束语 | 点线网面,一起构建MySQL知识网络
MySQL实战45讲
登录|注册

33 | 我查这么多数据,会不会把数据库内存打爆?

林晓斌 2019-01-28
我经常会被问到这样一个问题:我的主机内存只有 100G,现在要对一个 200G 的大表做全表扫描,会不会把数据库主机的内存用光了?
这个问题确实值得担心,被系统 OOM(out of memory)可不是闹着玩的。但是,反过来想想,逻辑备份的时候,可不就是做整库扫描吗?如果这样就会把内存吃光,逻辑备份不是早就挂了?
所以说,对大表做全表扫描,看来应该是没问题的。但是,这个流程到底是怎么样的呢?

全表扫描对 server 层的影响

假设,我们现在要对一个 200G 的 InnoDB 表 db1. t,执行一个全表扫描。当然,你要把扫描结果保存在客户端,会使用类似这样的命令:
mysql -h$host -P$port -u$user -p$pwd -e "select * from db1.t" > $target_file
你已经知道了,InnoDB 的数据是保存在主键索引上的,所以全表扫描实际上是直接扫描表 t 的主键索引。这条查询语句由于没有其他的判断条件,所以查到的每一行都可以直接放到结果集里面,然后返回给客户端。
那么,这个“结果集”存在哪里呢?
实际上,服务端并不需要保存一个完整的结果集。取数据和发数据的流程是这样的:
获取一行,写到 net_buffer 中。这块内存的大小是由参数 net_buffer_length 定义的,默认是 16k。
重复获取行,直到 net_buffer 写满,调用网络接口发出去。
如果发送成功,就清空 net_buffer,然后继续取下一行,并写入 net_buffer。
如果发送函数返回 EAGAIN 或 WSAEWOULDBLOCK,就表示本地网络栈(socket send buffer)写满了,进入等待。直到网络栈重新可写,再继续发送。
取消
完成
0/1000字
划线
笔记
复制
© 版权归极客邦科技所有,未经许可不得传播售卖。 页面已增加防盗追踪,如有侵权极客邦将依法追究其法律责任。
该试读文章来自付费专栏《MySQL实战45讲》,如需阅读全部文章,
请订阅文章所属专栏。
立即订阅
登录 后留言

精选留言(56)

  • 700 置顶
    老师,您好。根据文章内容,提炼如下信息:
    如果你看到 State 的值一直处于“Sending to client”,就表示服务器端的网络栈写满了。
    如何处理?
    1)使用 mysql_store_result 这个接口,直接把查询结果保存到本地内存。
    2)优化查询结果,并评估这么多的返回结果是否合理。
    3)而如果要快速减少处于这个状态的线程的话,将 net_buffer_length 参数设置为一个更大的值是一个可选方案。
    对于第3)方案不是很懂,“Sending to client” 表示服务器端的网路栈写满了,那不是应该加大 socket send buffer 吗?跟加大 net_buffer_length 有什么关系?net_buffer_length 加再大,但 socket send buffer 很小的话,网络栈不还是处于写满状态?

    作者回复: 好问题👍 很好的思考👍

    是这样的,net_buffer_length 的最大值是 1G,这个值比 socket send buffer大(一般是几M)

    比如假设一个业务,他的平均查询结果都是10M (当然这个业务有有问题,最终是要通过业务解决)

    但是如果我把net_buffer_length 改成10M,就不会有“Sending to client” 的情况。虽然网络栈还是慢慢发的,但是那些没发完的都缓存在net_buffer中,对于执行器来说,都是“已经写出去了”。

    2019-01-28
    16
  • 长杰
    遇到过一个场景,用mysqldump对业务db做逻辑备份保存在客户端,客户端是虚拟机,磁盘很快满了,导致server端出现sending to client状态,更糟糕的是业务db更新频繁,导致undo表空间变大,db服务堵塞,服务端磁盘空间不足。

    作者回复: 非常好,正是我要说明的一个场景呢,直接用你的例子放在下篇答疑部分哈

    2019-01-28
    17
  • IceGeek17
    老师你好,几个问题:

    按照文中所述,net_buffer是属于MySQL Server层的,在InnoDB引擎层,会使用buffer pool (以page的形式),也就是一个查询所占用的内存是: net_buffer + buffer pool里相关的page页
    是不是可以这么理解?

    当net_buffer写满,会调用网络接口发出去,net_buffer里的内容是如何发给socket send buffer的,
    是一行一行的扔给socket send buffer,还是把net_buffer 里的内容一下子全部扔给 socket send buffer ?

    文中说发送成功然后清空net_buffer, 这里net_buffer是如何清空的,是等net_buffer里的内容全部发送成功,然后一次性清理,还是发送成功一部分清理一部分?

    看了置顶的700问题和回复,几点疑问:
    对于一个查询,执行器拿到的所有结果,如果可以一次性放入net_buffer, 对于执行器来说是不是意味着“全都写出去了”,也就不会有 sending to client 状态?
    只有当查询的结果,不能够全部放入net_buffer,需要等net_buffer里的内容清空后再继续放入后续的结果,这时候状态才是显示 sending to client ?
    当查询结果可以全部放入net_buffer, 执行器也不管 net_buffer是否发送给 socket send buffer,都认为执行完了 ?
    是不是这么理解?

    对buffer pool,当通过LRU 淘汰数据页的时候,如果此时该页的内容是新的(也就是磁盘上的内容是老的),是不是需要强制先走一个刷脏页的流程,等脏页刷完了,然后才能淘汰该数据页?

    作者回复: 1. “是一行一行的扔给socket send buffer,还是把net_buffer 里的内容一下子全部扔给 socket send buffer ?”
    ---- net_buffer写满,一起发,然后清空net_buffer,组装下一批 。好问题
    2. 跟上一个问题同一个答案;
    3. “对于一个查询,执行器拿到的所有结果,如果可以一次性放入net_buffer, 对于执行器来说是不是意味着“全都写出去了”,也就不会有 sending to client 状态?” ----是的

    4. 是的
    5. 对,这个就是我们其他文章中介绍的,“带着邻居节点一起刷”的那个阶段。

    2019-02-14
    14
  • 清风
    net_buffer 应该是针对每个请求线程单独分配的,还是共享net_buffer . 我的理解应该是每个线程一块。mysql 可以根据最大请求连接数,能够算出来mysql 使用net_buffer 的总大小。同时如果mysql 占用的内存不大,也将影响到Mysql 能够处理连接连接数的大小。 不知道这种猜测是否准确。 后面那个改进型的LRU 算法真的非常好,就跟JVM 中年轻带 老年代的内存区域划分和淘汰机制一样。在做系统设计的时候可以把这种设计应用一下。

    作者回复: 你的理解是对的,每个线程(session)一个

    2019-03-30
    5
  • XXL
    请教老师一个问题,
    之前在开发工程中实际有碰到这样的业务,批量从MySQL中查询大量数据,每次通过限制起始+limit数量的来分批次查询,后来有同事推荐使用MySQL JDBC中的fetchSize()方法,不做分页通过一次大查询然后客户端流式读取来批量查询数据,这个内部原理是否就是文中所说的使用了mysql_use_result接口读一行处理一行实现的流式?或者也是mysql_store_result方式客户端边缓存边处理?请老师指教

    作者回复: 对,这种一般就是用mysql_use_result
    各有优劣吧

    一次性取的好处是,对服务端只全表,只扫描一遍;坏处是可能会出现大事务。

    一般更常见的做法是,分批取,然后每一批拿到最大的一个id(主键值)
    下一批查询的时候用 where Id > N 这种写法

    好问题

    2019-02-14
    5
  • 老杨同志
    本身是研发没过这种经历。猜一种吧
    如果客户端A性能慢,迟迟不去读取socket receive buffer,server端就不能发送,此时如果客户端A要读取的数据被其他线程频繁update,由于mvcc的实现,这个变更会记录到undo log,大量的日志会不会使io飙升?可能比较极端才会吧。如果此时客户端性能恢复,服务端要读取最新数据,并通过undo log计算较早的版本,是不是要也占用大量的cpu资源或者io资源?谢谢老师

    作者回复: 👍
    再考虑下都是update的情况 😆

    2019-01-28
    5
  • Long
    最近没时间看,今天终于补完了几天的课。
    2019-01-28
    5
  • 几近虚年
    老师,看了课文和留言,有个问题有点懵。
    Sending to client 状态,到底是体现了服务端的网络栈 socket receive buffer 写满了(客户端未及时读取),还是 net_buffer 满了,还有查询结果没有写入到 net_buffer 中?

    作者回复: 网络栈 socket receive buffer 写满了

    2019-02-20
    4
  • Leon📷
    如果客户端读结果不及时,会堵住 MySQL 的查询过程,但是不会把内存打爆。这个是指客户端的tcp滑动窗口处理没有及时确认,导致server端的网络协议栈没有多余的空间可以发送数据,导致server的处理线程停止从db读取数据发送给client,是这样理解吗

    作者回复: 对的

    2019-01-30
    4
  • 冰点18
    InnoDB改进的LRU算法,如果遇到连续两次的全表扫描,会不会就把young区的3/5给覆盖掉了?因为两次扫描时间间隔会超过一秒?

    作者回复: 会的

    2019-04-01
    3
  • 700
    老师,您好。感谢解答。
    接上个问题。
    Sending to client 是发生在下面哪个阶段的事件呢?
    1)是 “获取一行,写到 net_buffer 中。”
    2)还是“直到 net_buffer 写满,调用网络接口发出去。” //即数据从 net_buffer 发到 socket send buffer?
    3)还是“将 socket send buffer 的数据发送给 socket receive buffer”

    从您的回答“但是如果我把net_buffer_length 改成10M,就不会有“Sending to client” 的情况。”,我感觉应该是属于第1)阶段的事件。但感觉这又与您说的“Sending to client 表示的是服务器端的网络栈写满了”相矛盾。

    作者回复: 写net_buffer -- > net_buffer满了,调用网络接口发 -->发不出去
    这个是同一个调用链条呀

    “哪个阶段”没看懂,是同一个时刻

    2019-01-28
    3
  • 尘封
    如果一行数据超过了net buffer length的默认值16KB会如何?

    作者回复: 满16k就发给socket send buffer

    2019-01-28
    3
  • 进阶的码农
    如果客户端是在主机上呢

    作者回复: 你把他们当作两个单独的行为,只是刚好共同同一个文件系统而已

    2019-03-01
    2
  • 天使梦泪
    老师好,针对我上次问您的mysql缓存中的数据储存问题,您回答可以一直保存的,具体是怎么实现一直保存的(也不是储存在磁盘上,是使用的内存)?内存重启了之后,缓存不就也丢失了,是怎么做到持久化保存的,老师可以帮忙详细解答下么?

    作者回复: InnoDB 的是buffer pool,是在内存里。

    ”内存重启了之后,缓存不就也丢失了,是怎么做到持久化保存的,老师可以帮忙详细解答下么?“
    没有保存,重启就没有了,要访问的时候需要重新去磁盘读

    2019-01-29
    2
  • Sinyo
    @700 的置顶提问
    老师你说:"但是如果把 net_buffer_length 改成 10M,就不会有“Sending to client”的情况。虽然网络栈还是慢慢发的,但是那些没发完的都缓存在net_buffer中,对于执行器来说,都是“已经写出去了”。"

    假如数据量有1G,而如果要快速减少处于这个状态的线程的话,我们把net_buffer_length 从10M改成1G,快速减少的那部分操作是不是只有服务端发送到net_buffer的这部分?这样就不会有“Sending to client”的情况么?

    作者回复: 还是会显示为“Sending to client”,但是语句已经执行完了。
    不会占着资源(比如MDL读锁)

    2019-01-29
    2
  • PHP-SICUN
    老师,您好,有两个问题麻烦解惑一下
    1.扫描200G的表时数据会先放到InnoDB buffer pool,然后发送时在读取到net_buffer吗?
    2.如果是的话,异常情况导致socket send buffer被写满,是不是会出现InnoDB buffer中的某一页有可能出现读取后面的行时,超过1s,而被放到yong区域的情况?
    不知道这样表述或者理解的对吗

    作者回复: 1. 是 ,但是也不是“全部放到buffer pool以后”才发,读的时候是一个page 一个page 地读的
    2. 会,好在这个是“某一页”而已,量不大。 好问题

    很好的思考

    2019-01-29
    2
  • 天使梦泪
    老师,我有个问题不明白,mysql从缓存中取数据,缓存里的数据是怎么实现可以保存一段时间的?

    作者回复: “保存一段时间”是啥意思,LRU算法不是按照时间的哈,如果没人来淘汰,是可以一直保存的。

    2019-01-28
    2
  • 如明如月
    之前有特殊功能需要从主要业务库拉取指定范围的数据到另外同一个库的其他数据表的动作(insert into xxxxx select xxx from xxx 这种操作)数据量在万级或者十万级,对于这种操作,和本文讲的应该有些不同吧?能否帮分析一下这种场景的大致情况呢?或者有什么好的建议吗?

    作者回复: 嗯,这个不会返回结果到客户端,所以网络上不会有问题

    引擎内部的扫描机制是差不多的

    唯一不同是这个过程可能对原表有行锁(如果设置的是RR)

    万或者十万还好,是小数据,可以考虑拿到客户端再写回去,避免锁的问题


    2019-01-28
    2
  • Zzz
    林老师,有几个问题想请教以下:
    1、哪种查询语句下MySQL 是“边读边发的”的呢?对于order by这种语句肯定是需要先全部拿到内存再做排序处理最后返回结果。
    2、MySQL是怎么判断出可以“边读边发的”,是不是看下语句是否带order by这种关键字?
    3、我有办法知道该执行语句是否“边读边发的”吗?

    作者回复: 这三个问题其实是同一个

    “边读边发”的意思是,算出来的结果才能发
    像order by,得先排序得到结果,然后才发出去,如果读了数据直接发,那肯定不行,那是错误的结果。

    所以要排序了以后再发,这时候就需要中间数据结构,sort buffer

    2019-01-28
    2
  • 是我的海
    在线上遇到过一个问题:
    1.在慢日志中看到一条业务 sql 语句执行时间较长,但是自己在数据库中执行时瞬间返回结果,可能是什么原因?会是由于发送的数据多阻塞造成的么 ?(net_buffer_length太小 ?)
    2.慢查询语句到底是如何定义的?处于sending to client 状态,是不是也会统计等待写入net_buffer_length 的时间呢 ?
    2019-03-28
    1
收起评论
56
返回
顶部