Break Reference Cycles in VBA using simulated Weak Object References. In a Garbage Collected language, a Weak Reference is not counted and does not protect the referenced object from being collected by the Garbage Collector (GC) unlike a Strong Reference. In a non-GC language the weak references are counted separately from the strong ones (e.g. SWIFT language). VBA does not have GC nor weak references.
In order to be referenced by WeakReference, a VBA class must implement the IWeakable interface. A WeakReference instance saves the memory address of the targeted object (the address of the default interface - not the IWeakable interface) and later uses it to retrieve the target object using the LibMemory
library from the VBA-MemoryTools repository.
To be safe, all WeakReferences must be informed that the targeted Object has been terminated. Unfortunately, the Class_Terminate event is not part of the interface so it cannot be forced to do anything. Because too much boilerplate code would need to be added to all classes implementing IWeakable it is probably best to encapsulate all the logic inside a separate class called WeakRefInformer which is to be contained by the targeted class. The main idea is that by not exposing the contained WeakRefInformer object, it will surely go out of scope when the object implementing IWeakable is terminated.
A quick visual example. Consider a "parent" object containing 2 "child" objects pointing back through weak references and a 3rd "loose" weak reference. This would look like:
Just import the following code modules in your VBA Project:
You will also need:
- LibMemory from the submodules folder or you can try the latest version here
Note that LibMemory
is not available in the Zip download. If cloning via GitHub Desktop the submodule will be pulled automatically by default. If cloning via Git Bash then use something like:
$ git clone https://github.com/cristianbuse/VBA-WeakReference
$ git submodule init
$ git submodule update
or:
$ git clone --recurse-submodules https://github.com/cristianbuse/VBA-WeakReference
In all classes that need to be compatible with/referenced by WeakReference
, add the following code:
Implements IWeakable
Private Sub IWeakable_AddWeakRef(ByVal wRef As WeakReference)
Static informer As New WeakRefInformer
informer.AddWeakRef wRef, Me
End Sub
Create a new instance of the WeakReference class and assign the desired object:
Set wRef = New WeakReference
Set wRef.Object = targetObj 'targetObject implements IWeakable
Retrieving the object can be done at any time using:
Set obj = wRef.Object
Class1
:
'Class1
Option Explicit
Implements IWeakable
Public x As Long
Private Sub IWeakable_AddWeakRef(ByVal wRef As WeakReference)
Static informer As New WeakRefInformer
informer.AddWeakRef wRef, Me
End Sub
Method in a regular code module:
Sub TestWeakReference()
Dim c As Class1
Dim w1 As New WeakReference
Dim w2 As New WeakReference
Dim w3 As New WeakReference
'
Set c = New Class1
c.x = 1
'
Set w1.Object = c
Set w2.Object = c
Set w3.Object = c
Debug.Print w1.Object.x 'Prints 1 (correct)
Debug.Print w2.Object.x 'Prints 1 (correct)
Debug.Print w3.Object.x 'Prints 1 (correct)
Debug.Print TypeName(w1.Object) 'Prints Class1 (correct)
Debug.Print TypeName(w2.Object) 'Prints Class1 (correct)
Debug.Print TypeName(w3.Object) 'Prints Class1 (correct)
'
Dim temp As Class1
Set temp = New Class1
Set w3.Object = temp
temp.x = 2
'
Set c = Nothing 'Note this only resets w1 and w2 (not w3)
Set c = New Class1
c.x = 3
'
Debug.Print TypeName(w1.Object) 'Prints Nothing (correct)
Debug.Print TypeName(w2.Object) 'Prints Nothing (correct)
Debug.Print TypeName(w3.Object) 'Prints Class1 (correct)
Debug.Print w3.Object.x 'Prints 2 (correct)
End Sub
Import the following code modules from demo:
- DemoChild.cls
- DemoParent.cls
- DemoWeakRef.bas
- DemoChild2.cls
- DemoParent2.cls
and execute method:
DemoWeakRef.DemoMain
- There are no memory leaks even if state is lost.
- If the saved object has been destroyed, the WeakReference.Object (Get property) safely returns Nothing
- The WeakRefInformer.cls is not really needed but avoids the duplication of the same code across all classes implementing IWeakable. Just the minimal code presented above in the Usage section is needed when using the informer.
Many thanks to Greedo (GitHub / CR). See his contributions on CodeReview.
MIT License
Copyright (c) 2020 Ion Cristian Buse
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.