# Weak Reference class that allows a referenced object to be
# A WeakRef may be used exactly like the object it references.
# 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)
# With help from WeakRef, we can implement our own rudimentary WeakHash class.
# We will call it WeakHash, since it's really just a Hash except all of it's
# keys and values can be garbage collected.
# super WeakRef.new(key), WeakRef.new(obj)
# This is just a simple implementation, we've opened the Hash class and changed
# Hash#store to create a new WeakRef object with +key+ and +obj+ parameters
# before passing them as our key-value pair to the hash.
# With this you will have to limit your self to String keys, otherwise you
# will get an ArgumentError because WeakRef cannot create a finalizer for a
# Symbol. Symbols are immutable and cannot be garbage collected.
# Let's see it in action:
# #=> {"foo"=>"bar", "baz"=>#<Object:0x007f4ddfc6cb48>, "qux"=>"lol"}
# # Now run the garbage collector
# #=> WeakRef::RefError: Invalid Reference - probably recycled
# You can see the local variable +omg+ stayed, although its reference in our
# hash object was garbage collected, along with the rest of the keys and
# values. Also, when we tried to inspect our hash, we got a WeakRef::RefError.
# This is because these objects were also garbage collected.
class WeakRef < Delegator
# RefError is raised when a referenced object has been recycled by the
class RefError < StandardError
@@__map = ::ObjectSpace::WeakMap.new
# Creates a weak reference to +orig+
# Raises an ArgumentError if the given +orig+ is immutable, such as Symbol,
@@__map[self] or defined?(@delegate_sd_obj) ? @delegate_sd_obj :
Kernel::raise(RefError, "Invalid Reference - probably recycled", Kernel::caller(2))
def __setobj__(obj) # :nodoc:
# Returns true if the referenced object is still alive.
@@__map.key?(self) or defined?(@delegate_sd_obj)