liyujie
2025-08-28 d9927380ed7c8366f762049be9f3fee225860833
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
// run
 
// Copyright 2018 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
 
package main
 
import (
   "fmt"
   "runtime"
)
 
type HeapObj [8]int64
 
type StkObj struct {
   h *HeapObj
}
 
var n int
var c int = -1
 
func gc() {
   // encourage heap object to be collected, and have its finalizer run.
   runtime.GC()
   runtime.GC()
   runtime.GC()
   n++
}
 
func main() {
   f()
   gc() // prior to stack objects, heap object is not collected until here
   if c < 0 {
       panic("heap object never collected")
   }
   if c != 1 {
       panic(fmt.Sprintf("expected collection at phase 1, got phase %d", c))
   }
}
 
func f() {
   var s StkObj
   s.h = new(HeapObj)
   runtime.SetFinalizer(s.h, func(h *HeapObj) {
       // Remember at what phase the heap object was collected.
       c = n
   })
   g(&s)
   gc()
}
 
func g(s *StkObj) {
   gc() // heap object is still live here
   runtime.KeepAlive(s)
   gc() // heap object should be collected here
}