您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > Go语言

关于 runtime.KeepAlive() 你知道多少

时间:2020-10-19 12:15:03  来源:  作者:

我在看 go101 网站的 类型不安全指针[1](来源)[2] 一文时,偶然发现了runtime[3] 库的一个有趣的新方法 runtime.KeepAlive()[4] 的一个用法。刚开始我对于怎么使用它是很困惑的, 那么按我的性格肯定要探究它是怎么工作的。

runtime.KeepAlive 所做的事就是使一个变量保持 '存活',这就意味着它(或者它引用的变量)不会被垃圾收集,而且它所注册的任何终止器(finalizer)都不会被执行。这个文档[5] 中有一个如何使用它的例子。我的第一个疑问是为什么在代码中 runtime.KeepAlive() 的使用时机那么的靠后;我比较希望它能够更早的被调用,就像终止器被注入时,但是后来我明白了它这样做的真正意图。简而言之, runtime.KeepAlive() 是调用一下变量。显而易见的,一个变量直至它的最后一次使用期间都是存活的,所以如果你在后面使用一个变量,那么 Go 必须让它一直存活到最后使用的时候。

一方面,runtime.keepAlive 没有什么神奇的地方;任何一种使用某个变量的方式,都会使它保持存活。另一方面,runtime.KeepAlive() 是一种很重要的魔法,它表示 Go 保证了你所使用的变量不会被优化清除掉,因为编译器能明白没有什么能真正依赖于你的使用。虽然有很多其它的方式来使用一个变量,但即使是最聪明的方式也很容易受到编译器的影响,最聪明的方式也会有不利的一面,他们会影响 Go 的智能合理逃逸分析[6],强行将一个本属于本地栈的变量分配到堆上。

关于 runtime.KeepAlive() 的另一个特殊戏法是它的是实现方式,代码里什么都没做。实际上,它不是作为一个被调用的函数,而是由 ssa.go[7] 实现的编译器内部实现,类似于 unsafe.Pointer。当你的代码中使用了 runtime.KeepAlive(),Go 编译器会设置一个名为 OpKeepAlive 的静态单赋值(SSA),然后剩余的编译就会知道将这个变量的存活期保证到使用了 runtime.KeepAlive() 的时刻。

(阅读 ssa.go 的初始化函数是很有趣的。不出所料,有许多语义化包函数调用被直接映射到将指令内联在代码中,如 math.Sqrt。有些是平台相关的,包括 bits[8] 的函数)

runtime.KeepAlive() 是一个特别的魔法有一个直接的后果就是你不能得到它的地址。如果你这样做的话, Go 会报错:

./tst.go:20:22: cannot take the address of runtime.KeepAlive

我不知道 Go 是否会聪明地优化掉一个只调用 runtime.KeepAlive 的函数, 但希望你永远不需要间接调用 runtime.KeepAlive。

PS:尽管我很想说没有人应该需要对分配在栈上的本地变量(包括参数)调用 runtime.KeepAlive,因为在函数返回之前栈是不会被回收的,但这是一个危险的假设。编译器可以非常聪明地为两个不同的、没有重叠生存期的变量重用堆栈槽,或者简单地告诉垃圾收集它已经完成了某些工作(例如,用 nil 覆盖指向对象的指针)。


via: https://utcc.utoronto.ca/~cks/space/blog/programming/GoRuntimeKeepAliveNotes

作者:ChrisSiebenmann[9]译者:yuhang-dong[10]校对:unknwon[11]

本文由 GCTT[12] 原创编译,Go 中文网[13] 荣誉推出

参考资料

[1]

类型不安全指针: https://go101.org/article/unsafe.html

[2]

(来源): https://old.reddit.com/r/golang/comments/8ll6lf/how_to_safely_use_typeunsafe_pointers_in_go/

[3]

runtime: https://golang.org/pkg/runtime/

[4]

runtime.KeepAlive(): https://golang.org/pkg/runtime/#KeepAlive

[5]

这个文档: https://golang.org/pkg/runtime/#KeepAlive

[6]

Go 的智能合理逃逸分析: https://utcc.utoronto.ca/~cks/space/blog/programming/GoReflectEscapeHack

[7]

ssa.go: https://github.com/golang/go/blob/master/src/cmd/compile/internal/gc/ssa.go#L2828

[8]

bits: https://golang.org/pkg/math/bits/

[9]

ChrisSiebenmann: https://utcc.utoronto.ca/~cks/space/People/ChrisSiebenmann

[10]

yuhang-dong: https://github.com/yuhang-dong

[11]

unknwon: https://github.com/unknwon

[12]

GCTT: https://github.com/studygolang/GCTT

[13]

Go 中文网: https://studygolang.com/



Tags:runtime.KeepAlive()   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
我在看 go101 网站的 类型不安全指针[1](来源)[2] 一文时,偶然发现了runtime[3] 库的一个有趣的新方法 runtime.KeepAlive()[4] 的一个用法。刚开始我对于怎么使用它是很困惑...【详细内容】
2020-10-19  Tags: runtime.KeepAlive()  点击:(101)  评论:(0)  加入收藏
▌简易百科推荐
zip 是一种常见的归档格式,本文讲解 Go 如何操作 zip。首先看看 zip 文件是如何工作的。以一个小文件为例:(类 Unix 系统下)$ cat hello.textHello!执行 zip 命令进行归档:$ zip...【详细内容】
2021-12-17  Go语言中文网    Tags:Go语言   点击:(12)  评论:(0)  加入收藏
大家好,我是 polarisxu。前段时间,Russ Cox 明确了泛型相关的事情,原计划在标准库中加入泛型相关的包,改放到 golang.org/x/exp 下。目前,Go 泛型的主要设计者 ianlancetaylor 完...【详细内容】
2021-11-30  Go语言中文网    Tags:slices 包   点击:(24)  评论:(0)  加入收藏
前言最近因为项目需要写了一段时间的 Go ,相对于 Java 来说语法简单同时又有着一些 Python 之类的语法糖,让人大呼”真香“。 但现阶段相对来说还是 Python 写的多一些,偶尔还...【详细内容】
2021-11-25  crossoverJie    Tags:Go   点击:(29)  评论:(0)  加入收藏
go-micro是基于 Go 语言用于开发的微服务的 RPC 框架,主要功能如下:服务发现,负载均衡 ,消息编码,请求/响应,Async Messaging,可插拔接口,最后这个功能牛p安装步骤安装proto...【详细内容】
2021-09-06    石老师小跟班  Tags:go-micro   点击:(196)  评论:(0)  加入收藏
GoLand 2021.2 EAP 5 现已发布。用户可以从工具箱应用程序中获得 EAP 构建,也可以从官方网站手动下载。并且从此 EAP 开始,只有拥有有效的 JetBrains 帐户才能加入该计划。手...【详细内容】
2021-06-29  IT实战联盟  今日头条  Tags:GoLand   点击:(185)  评论:(0)  加入收藏
作者:HDT3213今天给大家带来的开源项目是 Godis:一个用 Go 语言实现的 Redis 服务器。支持: 5 种数据结构(string、list、hash、set、sortedset) 自动过期(TTL) 发布订阅、地理位...【详细内容】
2021-06-18  HelloGitHub  今日头条  Tags:Go   点击:(125)  评论:(0)  加入收藏
统一规范篇合理规划目录本篇主要描述了公司内部同事都必须遵守的一些开发规矩,如统一开发空间,既使用统一的开发工具来保证代码最后的格式的统一,开发中对文件和代码长度的控制...【详细内容】
2021-05-18  1024课堂    Tags:Go语言   点击:(232)  评论:(0)  加入收藏
闭包概述 闭包不是Go语言独有的概念,在很多编程语言中都有闭包 闭包就是解决局部变量不能被外部访问的一种解决方案 是把函数当作返回值的一种应用 代码演示总体思想:在函数...【详细内容】
2021-05-14  HelloGo  今日头条  Tags:Go语言   点击:(223)  评论:(0)  加入收藏
一时想不开,想了解一下Go语言,于是安装了并体验了一下。下载1. 进入golang.google.cn 点击Download Go 2.选择对应的操作系统,点击后开始下载。 安装1. windows下执行傻瓜式安...【详细内容】
2021-05-12  程序员fearlazy  fearlazy  Tags:Go语言   点击:(236)  评论:(0)  加入收藏
1.简介channel是Go语言的一大特性,基于channel有很多值得探讨的问题,如 channel为什么是并发安全的? 同步通道和异步通道有啥区别? 通道为何会阻塞协程? 使用通道导致阻塞的协程...【详细内容】
2021-05-10  程序员麻辣烫  今日头条  Tags:Go通道   点击:(272)  评论:(0)  加入收藏
相关文章
    无相关信息
最新更新
栏目热门
栏目头条