golang 源码剖析(8): 析构SetFinalizer

SetFinalizer设置一个finalizer关联到一个对象obj,当垃圾回收准备回收obj的时候,它会断开这个连接,并在单独的goroutine中执行finalizer(obj),这将让obj再次可用,但是不会再关联finalizer,
在SetFinalizer中:

  1. 先检查元素,如果类型为nil,或者不为指针,元素值为nil 都会报错
  2. 查找对象的heap地址
  3. 如果是函数类型为nil,则移除removefinalizer
  4. 检测参数类型,并给计算返回值size
  5. createfing()确保finalizer goroutine正在运行
  6. 添加finalizer
func SetFinalizer(obj interface{}, finalizer interface{}) {
	e := efaceOf(&obj)
	etyp := e._type
	if etyp == nil {
		throw("runtime.SetFinalizer: first argument is nil")
	}
	if etyp.kind&kindMask != kindPtr {
		throw("runtime.SetFinalizer: first argument is " + etyp.string() + ", not pointer")
	}
	ot := (*ptrtype)(unsafe.Pointer(etyp))
	if ot.elem == nil {
		throw("nil elem type!")
	}
	// find the containing object
	base, _, _ := findObject(uintptr(e.data), 0, 0)
	f := efaceOf(&finalizer)
	ftyp := f._type
	if ftyp == nil {
	// switch to system stack and remove finalizer
		systemstack(func() {
			removefinalizer(e.data)
		})
		return
    }
	fint := ft.in()[0]
	// make sure we have a finalizer goroutine
	createfing()

	systemstack(func() {
		if !addfinalizer(e.data, (*funcval)(f.data), nret, fint, ot) {
			throw("runtime.SetFinalizer: finalizer already set")
		}
	})
}

从mheap.specialfinalizeralloc中申请一块内存,然后调用addspecial,如果成功则返回,失败则free掉刚才申请的内存
addspecial ,根据p指针找到对应的span,轮训找到是否有相同offset和kind的数据存在,如果存在则finalizer函数已经存在,否则将其加入到span.specials中去(这里有根据offset排序)

// Adds a finalizer to
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值