blob: 7ae32441ef60703c4b336c7ac9f8afd8c9c16ce4 [file] [log] [blame]
Aurimas Liutikas93554f22022-04-19 16:51:35 -07001/*
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
17package android.util;
18
19import android.compat.annotation.UnsupportedAppUsage;
20
21/**
22 * Helper class for crating pools of objects. An example use looks like this:
23 * <pre>
24 * public class MyPooledClass {
25 *
26 * private static final SynchronizedPool<MyPooledClass> sPool =
27 * new SynchronizedPool<MyPooledClass>(10);
28 *
29 * public static MyPooledClass obtain() {
30 * MyPooledClass instance = sPool.acquire();
31 * return (instance != null) ? instance : new MyPooledClass();
32 * }
33 *
34 * public void recycle() {
35 * // Clear state if needed.
36 * sPool.release(this);
37 * }
38 *
39 * . . .
40 * }
41 * </pre>
42 *
43 * @hide
44 */
45public final class Pools {
46
47 /**
48 * Interface for managing a pool of objects.
49 *
50 * @param <T> The pooled type.
51 */
52 public static interface Pool<T> {
53
54 /**
55 * @return An instance from the pool if such, null otherwise.
56 */
57 @UnsupportedAppUsage
58 public T acquire();
59
60 /**
61 * Release an instance to the pool.
62 *
63 * @param instance The instance to release.
64 * @return Whether the instance was put in the pool.
65 *
66 * @throws IllegalStateException If the instance is already in the pool.
67 */
68 @UnsupportedAppUsage
69 public boolean release(T instance);
70 }
71
72 private Pools() {
73 /* do nothing - hiding constructor */
74 }
75
76 /**
77 * Simple (non-synchronized) pool of objects.
78 *
79 * @param <T> The pooled type.
80 */
81 public static class SimplePool<T> implements Pool<T> {
82 @UnsupportedAppUsage
83 private final Object[] mPool;
84
85 private int mPoolSize;
86
87 /**
88 * Creates a new instance.
89 *
90 * @param maxPoolSize The max pool size.
91 *
92 * @throws IllegalArgumentException If the max pool size is less than zero.
93 */
94 @UnsupportedAppUsage
95 public SimplePool(int maxPoolSize) {
96 if (maxPoolSize <= 0) {
97 throw new IllegalArgumentException("The max pool size must be > 0");
98 }
99 mPool = new Object[maxPoolSize];
100 }
101
102 @Override
103 @SuppressWarnings("unchecked")
104 @UnsupportedAppUsage
105 public T acquire() {
106 if (mPoolSize > 0) {
107 final int lastPooledIndex = mPoolSize - 1;
108 T instance = (T) mPool[lastPooledIndex];
109 mPool[lastPooledIndex] = null;
110 mPoolSize--;
111 return instance;
112 }
113 return null;
114 }
115
116 @Override
117 @UnsupportedAppUsage
118 public boolean release(T instance) {
119 if (isInPool(instance)) {
120 throw new IllegalStateException("Already in the pool!");
121 }
122 if (mPoolSize < mPool.length) {
123 mPool[mPoolSize] = instance;
124 mPoolSize++;
125 return true;
126 }
127 return false;
128 }
129
130 private boolean isInPool(T instance) {
131 for (int i = 0; i < mPoolSize; i++) {
132 if (mPool[i] == instance) {
133 return true;
134 }
135 }
136 return false;
137 }
138 }
139
140 /**
141 * Synchronized pool of objects.
142 *
143 * @param <T> The pooled type.
144 */
145 public static class SynchronizedPool<T> extends SimplePool<T> {
146 private final Object mLock;
147
148 /**
149 * Creates a new instance.
150 *
151 * @param maxPoolSize The max pool size.
152 * @param lock an optional custom object to synchronize on
153 *
154 * @throws IllegalArgumentException If the max pool size is less than zero.
155 */
156 public SynchronizedPool(int maxPoolSize, Object lock) {
157 super(maxPoolSize);
158 mLock = lock;
159 }
160
161 /** @see #SynchronizedPool(int, Object) */
162 @UnsupportedAppUsage
163 public SynchronizedPool(int maxPoolSize) {
164 this(maxPoolSize, new Object());
165 }
166
167 @Override
168 @UnsupportedAppUsage
169 public T acquire() {
170 synchronized (mLock) {
171 return super.acquire();
172 }
173 }
174
175 @Override
176 @UnsupportedAppUsage
177 public boolean release(T element) {
178 synchronized (mLock) {
179 return super.release(element);
180 }
181 }
182 }
183}