WeakRef
WeakRef 类
Parent:Delegator
弱引用(Weak Reference)类,允许引用的对象被垃圾回收。
WeakRef 可能与其引用的对象完全相同。
用法:
foo = Object.new # create a new object instance
p foo.to_s # original's class
foo = WeakRef.new(foo) # reassign foo with WeakRef instance
p foo.to_s # should be same class
GC.start # start the garbage collector
p foo.to_s # should raise exception (recycled)
示例
在 WeakRef 的帮助下,我们可以实现我们自己初级的 WeakHash 类。
我们将它称为 WeakHash,因为它实际上只是一个哈希,除了所有的键和值都可以被垃圾收集。
require 'weakref'
class WeakHash < Hash
def []= key, obj
super WeakRef.new(key), WeakRef.new(obj)
end
end
这只是一个简单的实现,我们已经打开了 Hash 类并更改了 Hash#store,以便在将它们作为我们的键值对传递给散列之前,使用key
和obj
参数创建一个新的 WeakRef 对象。
有了这个,你将不得不限制自己的字符串键,否则你会得到一个 ArgumentError,因为 WeakRef 无法为符号创建终结器。符号是不可改变的,不能被垃圾收集。
让我们在行动中看看它:
omg = "lol"
c = WeakHash.new
c['foo'] = "bar"
c['baz'] = Object.new
c['qux'] = omg
puts c.inspect
#=> {"foo"=>"bar", "baz"=>#<Object:0x007f4ddfc6cb48>, "qux"=>"lol"}
# Now run the garbage collector
GC.start
c['foo'] #=> nil
c['baz'] #=> nil
c['qux'] #=> nil
omg #=> "lol"
puts c.inspect
#=> WeakRef::RefError: Invalid Reference - probably recycled
你可以看到局部变量omg
保持不变,虽然它在我们的哈希对象中的引用是垃圾收集的,还有其他的键和值。另外,当我们试图检查我们的散列时,我们得到了一个 WeakRef :: RefError。这是因为这些对象也是垃圾收集的。
公共类方法
new(orig) 显示源代码
创建一个弱引用(weak reference)到 orig
如果给定orig
是不可变的,比如 Symbol,Integer 或 Float,则引发一个 ArgumentError 。
调用超类方法 Delegator.new
# File lib/weakref.rb, line 82
def initialize(orig)
case orig
when true, false, nil
@delegate_sd_obj = orig
else
@@__map[self] = orig
end
super
end
公共实例方法
weakref_alive?()显示源文件
如果引用的对象仍然存在,则返回 true。
# File lib/weakref.rb, line 103
def weakref_alive?
@@__map.key?(self) or defined?(@delegate_sd_obj)
end