1 /* 2 * Copyright (C) 2009 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.util; 18 19 /** 20 * Helper class for crating pools of objects. An example use looks like this: 21 * <pre> 22 * public class MyPooledClass { 23 * 24 * private static final SynchronizedPool<MyPooledClass> sPool = 25 * new SynchronizedPool<MyPooledClass>(10); 26 * 27 * public static MyPooledClass obtain() { 28 * MyPooledClass instance = sPool.acquire(); 29 * return (instance != null) ? instance : new MyPooledClass(); 30 * } 31 * 32 * public void recycle() { 33 * // Clear state if needed. 34 * sPool.release(this); 35 * } 36 * 37 * . . . 38 * } 39 * </pre> 40 * 41 * @hide 42 */ 43 public final class Pools { 44 45 /** 46 * Interface for managing a pool of objects. 47 * 48 * @param <T> The pooled type. 49 */ 50 public static interface Pool<T> { 51 52 /** 53 * @return An instance from the pool if such, null otherwise. 54 */ acquire()55 public T acquire(); 56 57 /** 58 * Release an instance to the pool. 59 * 60 * @param instance The instance to release. 61 * @return Whether the instance was put in the pool. 62 * 63 * @throws IllegalStateException If the instance is already in the pool. 64 */ release(T instance)65 public boolean release(T instance); 66 } 67 Pools()68 private Pools() { 69 /* do nothing - hiding constructor */ 70 } 71 72 /** 73 * Simple (non-synchronized) pool of objects. 74 * 75 * @param <T> The pooled type. 76 */ 77 public static class SimplePool<T> implements Pool<T> { 78 private final Object[] mPool; 79 80 private int mPoolSize; 81 82 /** 83 * Creates a new instance. 84 * 85 * @param maxPoolSize The max pool size. 86 * 87 * @throws IllegalArgumentException If the max pool size is less than zero. 88 */ SimplePool(int maxPoolSize)89 public SimplePool(int maxPoolSize) { 90 if (maxPoolSize <= 0) { 91 throw new IllegalArgumentException("The max pool size must be > 0"); 92 } 93 mPool = new Object[maxPoolSize]; 94 } 95 96 @Override 97 @SuppressWarnings("unchecked") acquire()98 public T acquire() { 99 if (mPoolSize > 0) { 100 final int lastPooledIndex = mPoolSize - 1; 101 T instance = (T) mPool[lastPooledIndex]; 102 mPool[lastPooledIndex] = null; 103 mPoolSize--; 104 return instance; 105 } 106 return null; 107 } 108 109 @Override release(T instance)110 public boolean release(T instance) { 111 if (isInPool(instance)) { 112 throw new IllegalStateException("Already in the pool!"); 113 } 114 if (mPoolSize < mPool.length) { 115 mPool[mPoolSize] = instance; 116 mPoolSize++; 117 return true; 118 } 119 return false; 120 } 121 isInPool(T instance)122 private boolean isInPool(T instance) { 123 for (int i = 0; i < mPoolSize; i++) { 124 if (mPool[i] == instance) { 125 return true; 126 } 127 } 128 return false; 129 } 130 } 131 132 /** 133 * Synchronized pool of objects. 134 * 135 * @param <T> The pooled type. 136 */ 137 public static class SynchronizedPool<T> extends SimplePool<T> { 138 private final Object mLock; 139 140 /** 141 * Creates a new instance. 142 * 143 * @param maxPoolSize The max pool size. 144 * @param lock an optional custom object to synchronize on 145 * 146 * @throws IllegalArgumentException If the max pool size is less than zero. 147 */ SynchronizedPool(int maxPoolSize, Object lock)148 public SynchronizedPool(int maxPoolSize, Object lock) { 149 super(maxPoolSize); 150 mLock = lock; 151 } 152 153 /** @see #SynchronizedPool(int, Object) */ SynchronizedPool(int maxPoolSize)154 public SynchronizedPool(int maxPoolSize) { 155 this(maxPoolSize, new Object()); 156 } 157 158 @Override acquire()159 public T acquire() { 160 synchronized (mLock) { 161 return super.acquire(); 162 } 163 } 164 165 @Override release(T element)166 public boolean release(T element) { 167 synchronized (mLock) { 168 return super.release(element); 169 } 170 } 171 } 172 } 173