您现在的位置是:亿华云 > 系统运维
详解内存管理机制的变更,你需要了解
亿华云2025-10-03 22:06:20【系统运维】3人已围观
简介本文转载自微信公众号「脑子进煎鱼了」,作者陈煎鱼。转载本文请联系脑子进煎鱼了公众号。大家好,我是正在学习如何蒸鱼的煎鱼。在上一篇 Go1.16 特性介绍的文章中我们有提到,从 v1.16 起,Go 在
本文转载自微信公众号「脑子进煎鱼了」,详解作者陈煎鱼。内存转载本文请联系脑子进煎鱼了公众号。管理更需
大家好,机制解我是详解正在学习如何蒸鱼的煎鱼。
在上一篇 Go1.16 特性介绍的内存文章中我们有提到,从 v1.16 起,管理更需Go 在 Linux 下的机制解默认内存管理策略会从MADV_FREE 改回 MADV_DONTNEED 策略。
这时候可能至少分两拨小伙伴,详解分别是内存:
知道是什么,被这个问题 “折磨“ 过的管理更需,瞬间眼前一亮。机制解 不知道是详解什么,出现了各种疑惑了,内存这说的管理更需都是些什么。灵魂拷问
你有没有以下的疑问,或者是否清楚:
文中所说的 MADV_FREE 是什么? 文中所说的 MADV_DONTNEED 是什么? 为什么特指 Go 语言的 Linux 环境? 为什么是站群服务器说从 MADV_FREE改回 MADV_DONTNEED?在今天这篇文章中我们都将进一步的展开和说明,让我们一同来了解这个改来改去的内存机制到底是何物。
madvise 爱与恨
在 Linux 系统中,在 Go Runtime 中通过系统调用 madvise(addr, length, advise) 方法,能够告诉内核如何处理从 addr 开始的 length 字节。
重点之一就是 ”如何处理“,在 Linux 下 Go 语言中目前支持两种策略,分别是(via @felix021):
MADV_FREE:内核会在进程的页表中将这些页标记为 “未分配”,从而进程的 RSS 就会变小。OS 后续可以将对应的物理页分配给其他进程。 MADV_DONTNEED:内核只会在页表中将这些进程页面标记为可回收,在需要的时候才回收这些页面。所带来的影响
Go 语言官方恰好就在 2019 年的 Go1.12 做了如下调整。
Go1.12 以前。 Go.12-Go1.15.Go1.12 以前
Go Runtime 在 Linux 上默认使用的是 MADV_DONTNEED 策略。
// 没有任何奇奇怪怪的判断 madvise(v, n, _MADV_DONTNEED)从整体效果来看,进程 RSS 可以下降的比较快,网站模板但从性能效率上来看差点。
Go1.12-Go1.15
当前 Linux 内核版本 >=4.5 时,Go Runtime 在 Linux 上默认使用了性能更为高效的 MADV_FREE 策略。
var advise uint32 if debug.madvdontneed != 0 { advise = _MADV_DONTNEED } else { advise = atomic.Load(&adviseUnused) } if errno := madvise(v, n, int32(advise)); advise == _MADV_FREE && errno != 0 { // MADV_FREE was added in Linux 4.5. Fall back to MADV_DONTNEED if it is // not supported. atomic.Store(&adviseUnused, _MADV_DONTNEED) madvise(v, n, _MADV_DONTNEED) }从整体效果来看,进程RSS 不会立刻下降,要等到系统有内存压力了才会释放占用,RSS 才会下降。
带来的副作用
故事往往不是那么的美好,显然在 Go1.12 起针对 madvise 的 MADV_FREE 策略的调整非常 “片面”。
来自社区微信群的小伙伴
结合社区里所遇到的案例可得知,该次调整带来了许多问题:
引发用户体验的问题:Go issues 上总是出现以为内存泄露,但其实只是未满足条件,内存没有马上释放的案例。 混淆统计信息和监控工具的情况:在 Grafana 等监控上,发现容器进程内存较高,释放很慢,告警了,很慌。 导致与内存使用有关联的个别管理系统集成不良:例如 Kubernetes HPA ,云服务器提供商或者自定义了扩缩容策略这类模式,难以评估。 挤压同主机上的其他应用资源:并不是所有的 Go 程序都一定独立跑在单一主机中,自然就会导致同一台主机上的其他应用受到挤压,这是难以评估的。从社区反馈来看是问题多多,弊大于利。
官方本想着想着性能更好一些,但是在现实场景中引发了不少的新问题,甚至有提到和 Android 流程管理不兼容的情况。
有种 “捡了芝麻,丢了西瓜” 的感觉。
Go1.16:峰回路转
既然社区反馈的问题何其多,有没有人提呢?有,超级多。
多到提出修改回 MADV_DONTNEED 的 issues 仅花了 1-2 天的时间就讨论完毕。
很快得出结论且合并 CL 关闭 issues 了。
Go1.16 修改内容如下:
func parsedebugvars() { // defaults debug.cgocheck = 1 debug.invalidptr = 1 if GOOS == "linux" { debug.madvdontneed = 1 } ... }直接指定回了 debug.madvdontneed = 1,简单粗暴。
总结
在本篇文章中,我们针对 Go 语言在 Linux 下的 madvise 方法的策略调整进行了历史介绍和说明,同时针对其调整所带来的的副作用及应对措施进行了一一介绍。
本次变更很好的印证了,牵一发动全身的说法。大家在后续应用这块时也要多加注意。
你觉得 Go1.16 这个特性变更怎么样呢?欢迎在评论区留言。
参考
runtime: default to MADV_DONTNEED on Linux
踩坑记:go 服务内存暴涨
Go 1.12 关于内存释放的一个改进
很赞哦!(76377)
相关文章
- 2、根据用户基础选择访问提供程序。由于互联问题的存在,接入商的选择也非常重要,如果用户群主要在联通,尽量选择联通接入较好的接入商,如果用户群主要在电信,那么选择电信接入较好的接入商。如果用户组位于国家/地区,则选择更好的访问提供程序进行交互。
- 22个Vue 源码中的工具函数
- 域名综合查询可以查询到哪些信息?
- Spring Boot 配置属性类型安全你都了解了吗?
- 用户邮箱的静态密码可能已被钓鱼和同一密码泄露。在没有收到安全警报的情况下,用户在适当的时间内不能更改密码。在此期间,攻击者可以随意输入帐户。启用辅助身份验证后,如果攻击者无法获取移动电话动态密码,他将无法进行身份验证。这样,除非用户的电子邮件密码和手机同时被盗,否则攻击者很难破解用户的邮箱。
- 千万级用户系统的SQL调优实战
- HTML标签和说明
- 面试突击:请求转发和请求重定向有什么区别?你懂了吗?
- 4、选择一个安全的域名注册商进行域名注册
- 十进制浮点数的表示方法
站长推荐
5、企业注册国内域名需要证件,其它情况一律不需要证件。
个人申请域名操作流程和注意事项
Java record vs lombok
Keepalived 高可用的三种路由方案
当投资者经过第二阶段的认真学习之后又充满了信心,认为自己可以在市场上叱咤风云地大干一场了。但没想到“看花容易绣花难”,由于对理论知识不会灵活运用.从而失去灵活应变的本能,就经常会出现小赢大亏的局面,结果往往仍以失败告终。这使投资者很是困惑和痛苦,不知该如何办,甚至开始怀疑这个市场是不是不适合自己。在这种情况下,有的人选择了放弃,但有的意志坚定者则决定做最后的尝试。
基于 Redis 构建简单分布式锁的局限
Go 语言内存逃逸案例
Python和Excel的完美结合:常用操作汇总(案例详析)