alt.hn

3/28/2025 at 6:58:00 AM

Finalizers are tricker than you might think. Part 2

https://sergeyteplyakov.github.io/Blog/2025/03/27/Finalizers_are_tricker_than_you_might_think_p2.html

by GOPbIHbI4

3/31/2025 at 5:31:54 AM

The same happens in ARC code as well. People expect scope-based lifetimes and are surprised to learn about “last use” rules.

by saagarjha

3/31/2025 at 6:59:31 AM

Those that don't keep up with WWDC talks are going to be even more surprised, given the changes on how ARC works across Swift versions.

Going back to finalizers, while they seemed a great idea since Smalltalk days, they have proven otherwise, and that is why nowadays they are largely deprecated.

In Java, and as of last Go's version, there are alternative ways with classical finalizers being "deprecated for removal" in Java's case, and on .NET SafeHandles and IDispose pattern is the way.

by pjmlp

3/31/2025 at 7:45:55 AM

Java has had PhantomReferce for about 25y now. There are meant for post-mortrem clean up. (although both finalize and PhantomReferce are pre-mortrem, unlike WeakRefences)

One of the main issues with finalize method is that it adds the newly created objects (prior to invoking the c-tor) to a (effectively) global queue which has rather negative perf. impact. The happens-before semantics also work in a weird way, e.g. fields set in the c-tor may not be visible in finalize().

Finalize/clean up code is still needed to ensure proper freeing of non-managed 'objects'. They are more of a defense mechanism in most cases as close/dispose is the correct way.

by xxs

3/31/2025 at 10:12:53 AM

PhantomReference solves the problem where objects can be resurrected during finalization, not the problem where objects can be finalized while you're still "using" them.

by immibis

3/31/2025 at 1:01:58 PM

>finalized while you're still "using" them.

It's a well known way to deal with such cases by using an additional weak reference. (even I think J. Bloch has mentioned it). Here (typing w/o an other editor, excuse typos):

  class X<T> extends PhantomRefenece<T> {
    final WeakReference<T> weak;
    X(T ref, ReferenceQueue<? super T> q){
      super(ref, q);
      this.weak = new WeakReference<>(ref);
    } 
  }

  ... 
  for (X x; (x = (X)q.poll()) != null;) {
    process(x.weak.get());
  }
This is why a mentioned it is effectively pre-mortrem. PhantomReference still keeps the referent, it's just not possible to retrieve it as get() is overridden.

by xxs

4/1/2025 at 8:22:44 AM

Bad idea. Weak reference can be removed at the same time. You're supposed to let the PhantomReference contain the underlying thing that you want to delete when the object is GCed, such as a native handle.

by immibis

3/31/2025 at 11:37:59 AM

Even orthography is trickier than one might think, as the title demonstrates.

by zombot