登录
首页 >  Golang >  Go问答

深入了解GoLang中的垃圾收集机制

来源:stackoverflow

时间:2024-03-26 17:09:37 424浏览 收藏

在 Go 语言中,垃圾收集器会释放不再被程序引用的内存。对于给定的类型 t,如果存在对该类型的引用,则其中包含的数据将不会被释放。在本文中,作者解释了如何使用转义分析来确定内存是在堆栈还是堆上分配的。通过分析示例代码,作者说明了当数据被传递给 t.Data 字段时,它将被 t 引用,因此只有当 t 不再被引用时才会被垃圾收集。

问题内容

我对 golang 的垃圾收集器有点困惑。

考虑下面的代码,我在其中为我的类型 t 实现了读取器接口。

type t struct {
   header header
   data   []*mydatatype
}

func (t *t) read(p []byte) (int, error) {
    t.header = *(*header) (t.readfileheader(p))
    t.data = *(*[]*mydatatype) (t.readfiledata(p))
}

在读取器函数中,我将使用 unsafe.pointer 将数据转换为 mydatatype,该指针将指向使用反射模块创建的切片(这更复杂,但对于示例来说这应该足够了)

func (t *t) readfiledata(data []byte, idx int, ...) unsafe.pointer {
    ...
    return unsafe.pointer(&reflect.sliceheader{data : uintptr(unsafe.pointer(&data[idx])), ...}) 
}

如果我要读取不同函数中的数据

func (d *Dummy) foo() {
    data, _ := ioutil.ReadFile(filename)
    d.t.Read(data) <---will GC free data?
}

现在我很困惑,gc 是否可能在退出 foo 函数后从文件中释放加载的数据。或者释放d.t后数据也会被释放。


解决方案


要了解 GC 可能会对变量执行什么操作,首先您需要知道 Go 如何以及在何处分配它们。这是关于 escape analysis 的好读物,这就是 Go 编译器如何决定在堆栈或堆之间分配内存的方式。

长话短说,只有当 Go 程序没有引用内存时,GC 才会释放内存。

在您的示例中,对 data, _ := ioutil.ReadFile(filename) 加载的数据的引用最终传递给 t.Data = *(*[]*MyDataType) (t.readFileData(p)) 。因此,只要 (t *T) 结构体也被引用,它们就会被引用。据我从您的代码中看到,加载的数据将与 (t *T) 一起被垃圾收集。

理论要掌握,实操不能落!以上关于《深入了解GoLang中的垃圾收集机制》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>