/************************************************************** * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. * *************************************************************/ package util; import java.lang.ref.PhantomReference; import java.lang.ref.ReferenceQueue; /** * Wait until an object has become unreachable. * *

Instances of this class will typically be used as either:

*
 *   SomeType o = new SomeType(...);
 *   ... // use "o"
 *   WaitUnreachable u = new WaitUnreachable(o);
 *   o = null;
 *   u.waitUnreachable();
 * 
*

or:

*
 *   WaitUnreachable u = new WaitUnreachable(new SomeType(...));
 *   ... // use "(SomeType) u.get()"
 *   u.waitUnreachable();
 * 
*/ public final class WaitUnreachable { /** * Creates a new waiter. * * @param obj the object on which to wait */ public WaitUnreachable(Object obj) { this.obj = obj; queue = new ReferenceQueue(); ref = new PhantomReference(obj, queue); } /** * Gets the object on which to wait. * * @return the object on which to wait, or null if * waitUnreachable has already been called */ public synchronized Object get() { return obj; } /** * Starts waiting for the object to become unreachable. * *

This blocks the current thread until the object has become * unreachable.

* *

Actually, this method waits until the JVM has detected that * the object has become unreachable. This is not deterministic, but this * methods makes a best effort to cause the JVM to eventually detect the * situation. With a typical JVM, this should suffice.

*/ public void waitUnreachable() { synchronized (this) { obj = null; } System.out.println("waiting for gc"); while (queue.poll() == null) { System.gc(); System.runFinalization(); byte[] dummy = new byte[1024]; } } /** * Ensures that an object will be finalized as soon as possible. * *

This does not block the current thread. Instead, a new thread is * spawned that busy waits for the given object to become unreachable.

* *

This method cannot guarantee that the given object is eventually * finalized, but it makes a best effort. With a typical JVM, this should * suffice.

* * @param obj the object of which to ensure finalization */ public static void ensureFinalization(final Object obj) { final class WaitThread extends Thread { public WaitThread(Object obj) { super("ensureFinalization"); unreachable = new WaitUnreachable(obj); } public void run() { unreachable.waitUnreachable(); } private final WaitUnreachable unreachable; } new WaitThread(obj).start(); } private Object obj; private final ReferenceQueue queue; private final PhantomReference ref; }