10.8 垃圾回收和 SetFinalizer

Go 开发者不需要写代码来释放程序中不再使用的变量和结构占用的内存,在 Go 运行时中有一个独立的进程,即垃圾收集器 (GC),会处理这些事情,它搜索不再使用的变量然后释放它们的内存。可以通过 runtime 包访问 GC 进程。

通过调用 runtime.GC() 函数可以显式的触发 GC,但这只在某些罕见的场景下才有用,比如当内存资源不足时调用 runtime.GC(),它会在此函数执行的点上立即释放一大片内存,此时程序可能会有短时的性能下降(因为 GC 进程在执行)。

如果想知道当前的内存状态,可以使用:

  1. // fmt.Printf("%d\n", runtime.MemStats.Alloc/1024)
  2. // 此处代码在 Go 1.5.1下不再有效,更正为
  3. var m runtime.MemStats
  4. runtime.ReadMemStats(&m)
  5. fmt.Printf("%d Kb\n", m.Alloc / 1024)

上面的程序会给出已分配内存的总量,单位是 Kb。进一步的测量参考 文档页面

如果需要在一个对象 obj 被从内存移除前执行一些特殊操作,比如写到日志文件中,可以通过如下方式调用函数来实现:

  1. runtime.SetFinalizer(obj, func(obj *typeObj))

func(obj *typeObj) 需要一个 typeObj 类型的指针参数 obj,特殊操作会在它上面执行。func 也可以是一个匿名函数。

在对象被 GC 进程选中并从内存中移除以前,SetFinalizer 都不会执行,即使程序正常结束或者发生错误。

练习 10.17 main_stack.go

从练习 10.16 开始(它基于结构体实现了一个栈结构),为栈的实现 (stack_struct.go) 创建一个单独的包 stack,并从 mainmain.stack.go 中调用它。

链接