/* * Copyright (C) 2005 The Guava Authors * * Licensed 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 com.google.common.base; import com.google.common.base.internal.Finalizer; import junit.framework.TestCase; import java.lang.ref.ReferenceQueue; import java.lang.ref.WeakReference; import java.net.URL; import java.net.URLClassLoader; /** * Unit test for {@link FinalizableReferenceQueue}. * * @author Bob Lee */ public class FinalizableReferenceQueueTest extends TestCase { private FinalizableReferenceQueue frq; @Override protected void tearDown() throws Exception { frq = null; } public void testFinalizeReferentCalled() { MockReference reference = new MockReference( frq = new FinalizableReferenceQueue()); // wait up to 5s for (int i = 0; i < 500; i++) { if (reference.finalizeReferentCalled) { return; } try { System.gc(); Thread.sleep(10); } catch (InterruptedException e) { /* ignore */ } } fail(); } static class MockReference extends FinalizableWeakReference { volatile boolean finalizeReferentCalled; MockReference(FinalizableReferenceQueue frq) { super(new Object(), frq); } @Override public void finalizeReferent() { finalizeReferentCalled = true; } } /** * Keeps a weak reference to the underlying reference queue. When this * reference is cleared, we know that the background thread has stopped * and released its strong reference. */ private WeakReference> queueReference; public void testThatFinalizerStops() { weaklyReferenceQueue(); // wait up to 5s for (int i = 0; i < 500; i++) { if (queueReference.get() == null) { return; } try { System.gc(); Thread.sleep(10); } catch (InterruptedException e) { /* ignore */ } } fail(); } /** * If we don't keep a strong reference to the reference object, it won't * be enqueued. */ FinalizableWeakReference reference; /** * Create the FRQ in a method that goes out of scope so that we're sure * it will be reclaimed. */ private void weaklyReferenceQueue() { frq = new FinalizableReferenceQueue(); queueReference = new WeakReference>(frq.queue); /* * Queue and clear a reference for good measure. We test later on that * the finalizer thread stopped, but we should test that it actually * started first. */ reference = new FinalizableWeakReference(new Object(), frq) { @Override public void finalizeReferent() { reference = null; frq = null; } }; } public void testDecoupledLoader() { FinalizableReferenceQueue.DecoupledLoader decoupledLoader = new FinalizableReferenceQueue.DecoupledLoader() { @Override URLClassLoader newLoader(URL base) { return new DecoupledClassLoader(new URL[] { base }); } }; Class finalizerCopy = decoupledLoader.loadFinalizer(); assertNotNull(finalizerCopy); assertNotSame(Finalizer.class, finalizerCopy); assertNotNull(FinalizableReferenceQueue.getStartFinalizer(finalizerCopy)); } static class DecoupledClassLoader extends URLClassLoader { public DecoupledClassLoader(URL[] urls) { super(urls); } @Override protected synchronized Class loadClass(String name, boolean resolve) throws ClassNotFoundException { // Force Finalizer to load from this class loader, not its parent. if (name.equals(Finalizer.class.getName())) { Class clazz = findClass(name); if (resolve) { resolveClass(clazz); } return clazz; } return super.loadClass(name, resolve); } } public void testGetFinalizerUrl() { assertNotNull(getClass().getResource("internal/Finalizer.class")); } }