it编程 > 数据库 > Mysql

MYSQL查询结果实现发送给客户端

8人参与 2025-06-11 Mysql

mysql取数据和发数据的流程(边读边发)

注意net_buffer满了就发给客户端而不是等查询语句查询完所有数据才发,其中net_buffer默认大小是16k,所以服务器占用内存最多是16k,不会因为查询200g的数据导致内存爆掉。

如果客户端接收数据太慢,就会导致数据在socket send buffer 堆积,导致发送数据到客户端的线程进入等待,由于net_buffer数据没有发送出去,net_buffer一直没法清空,查询语句就要等待net_buffer清空才能继续执行,导致服务端查询速度变慢。

sending to client

假设使用show processlist发现一条语句的state是“sending to client”,就是上面说的在等待socket send buffer 有足够空间可以将net_buffer中的数据写进去发送给客户端。

sending data

一个查询语句的状态是这样的:

但是mysql8好像改了,我看到的是executing

lru(least recently used )算法

最近最少使用算法是一种内存淘汰机制,核心是--内存空间不足时,将最少使用的数据页淘汰

innodb管理buffer pool 的lru算法,是通过链表来实现的,每个链表节点就是一个数据页。

但是这个lru在某些场景时有很大问题的

比如一个语句查询了历史数据表(非业务热点数据表)的所有数据,这就会把链表中之前维护的热点数据页都淘汰掉,留下一些非业务热点数据页,会导致缓存命中率急剧降低,磁盘io压力倍增。

针对这种清空,innodb对lru算法进行了优化

就是对链表进行了分区,前5/8是young区,后3/8是old区。

如果该数据页在old待的时间大于1秒,就移动到整个链表的头节点。

如果小于1秒,就不改变位置。

这个1秒是由innodb_old_blocks_time参数控制的。

这样一来,短时间内的非热点数据只会存在于old区,一段时间内不被访问就慢慢掉到尾部被淘汰了,不会影响到真正的热点数据,它们会一直在young区。保证了不影响正常业务。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持代码网。

(0)

您想发表意见!!点此发布评论

推荐阅读

MySQL 设置AUTO_INCREMENT 无效的问题解决

06-11

Mysql 中的日期时间函数示例详解

06-11

六个案例搞懂mysql间隙锁

06-11

杀死MySQL进程的多种方法实现

06-11

MySQL 8.0找不到 my.ini 配置文件(并开启 Binlog 监听)

06-11

MySQL 横向衍生表(Lateral Derived Tables)的实现

06-11

猜你喜欢

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论