“This is the third day of my participation in the Gwen Challenge in November. Check out the details: The last Gwen Challenge in 2021.”

scenario

A common problem encountered in programs is to create a large number of objects in a short period of time, resulting in memory constraints that trigger GC and lead to performance problems. For this problem, we can use object pooling technology to solve it. We need to understand the Android garbage collection mechanism before using it.

Garbage collection in Android

The most recently allocated object will be stored in the Young Generation area. When the object stays in the area for a certain amount of time, it will be moved to the Old Generation. Finally, go to the Permanent Generation area. Each level of memory area has a fixed size, and new objects are constantly allocated to this area. When the total size of these objects approaches the threshold of this level of memory area, the GC is triggered to make room for other new objects. Every time a GC occurs, all threads are suspended. It also depends on which Generation it is. Young Generation has the shortest time per GC operation, followed by Old Generation, and Permanent Generation has the longest.

The reason why GC fires frequently

  • A large number of objects are created and released in a short amount of time

  • Generating a large number of objects in an instant can severely consume the Young Generation’s memory area and trigger GC when it reaches a threshold where there is not enough free space. Even if the objects allocated each time take up a small amount of memory, they stack up to increase the Heap pressure, triggering more other types of GC. This operation can affect frame rates and cause perceived performance problems for users.

Object pooling

Work Project:

  • Pool represents an object Pool for storing reusable objects
  • The second step is to fetch the object.
  • The third step is to use the fetched object to complete some tasks. There is no need for the object pool to do anything, but it also means that the object is rented for a period of time and cannot be lent out by other components.
  • The fourth step is return. The component returns the loaned object so that it can continue to fulfill other loan requests.

Android object pool source code analysis

Public final class Pools {public static interface Pool<T> {/** * @return Fetch objects from object Pool */ @unsupportedAPpusage public T acquire(); * @return true True Indicates that the released object is successfully added to the object pool. ** @throws IllegalStateException If the object already exists, Public Boolean release(T instance); } private Pools() {/* do nothing-hiding constructor */ / public static class SimplePool<T> implements Pool<T> { @UnsupportedAppUsage private final Object[] mPool; Private int mPoolSize; private int mPoolSize; ** @param maxPoolSize Specifies the maximum capacity of the object pool ** / @unsupportedappusage public SimplePool(int  maxPoolSize) { if (maxPoolSize <= 0) { throw new IllegalArgumentException("The max pool size must be > 0"); } mPool = new Object[maxPoolSize]; } @override @suppressWarnings ("unchecked") @unsupportedappusage public T acquire() {if  (mPoolSize > 0) { final int lastPooledIndex = mPoolSize - 1; T instance = (T) mPool[lastPooledIndex]; mPool[lastPooledIndex] = null; mPoolSize--; return instance; } return null; } @override @unsupportedAPPUSage public Boolean release(T instance) {if (isInPool(instance)) { throw new IllegalStateException("Already in the pool!" ); } if (mPoolSize < mPool.length) { mPool[mPoolSize] = instance; mPoolSize++; return true; } return false; } private Boolean isInPool(T instance) {for (int I = 0; i < mPoolSize; i++) { if (mPool[i] == instance) { return true; } } return false; ** @param <T> The pooled type. */ public static class SynchronizedPool<T> extends SimplePool<T> { private final Object mLock; /** * Creates a new instance. ** @param maxPoolSize The Max pool size. * @param lock an optional custom object  to synchronize on * * @throws IllegalArgumentException If the max pool size is less than zero. */ public SynchronizedPool(int maxPoolSize, Object lock) { super(maxPoolSize); mLock = lock; } /** @see #SynchronizedPool(int, Object) */ @UnsupportedAppUsage public SynchronizedPool(int maxPoolSize) { this(maxPoolSize, new Object()); } @Override @UnsupportedAppUsage public T acquire() { synchronized (mLock) { return super.acquire(); } } @Override @UnsupportedAppUsage public boolean release(T element) { synchronized (mLock) { return super.release(element); }}}}Copy the code

Acquire (): a function that requests objects from the object Pool. Release (): a function that releases objects back into the object Pool

The SimplePool class principle: Uses the idea of lazy loading. When SimplePool initializes, no N objects of type T are generated and stored in the object pool. Instead, the released T-type objects are stored in the object pool every time an external call to release() is made. You put it in before you take it out.

Advantages and disadvantages of object pooling

advantages
  • Reuse objects in the object pool can avoid frequent creation and destruction of pairs in the heap, thus reducing the burden of garbage collector and reducing memory jitter.
  • You don’t have to initialize object state repeatedly, which is great for time-consuming constructors;
disadvantages
  • Object allocation in Java is no slower than malloc calls in C, and the overhead of allocating/releasing objects is negligible for light and medium magnitude objects.

  • In a concurrent environment, multiple threads may need to acquire objects from the pool, need to synchronize on the heap data structure or block due to lock contention, which can be hundreds of times more expensive than creating and destroying objects.

  • Due to the limited number of objects in the pool, it becomes a scalability bottleneck;

  • It is difficult to set the correct size of the object pool. If it is too small, it does not work; if it is too large, it consumes too much memory.

    Optimization: It is possible to start a thread for periodic scan analysis to compress the object pool to a suitable size to save memory, but in order to get good analysis results, reuse may need to be paused during the scan to avoid interference, resulting in inefficiency, or use very complex algorithm strategies (increasing maintenance difficulty);

  • The design and use of object pool are prone to errors, and state synchronization needs to be paid attention to in the design, which is a difficulty. In the use of object pool, there may be forgetting to return (just like forgetting free in C programming language), and repeated return (it may need to do a loop to determine whether the object exists in the pool, which is also a cost). Problems such as using objects after they are returned (potentially causing multiple threads to use an object concurrently).

Demo in source code

Because object pools are designed to be put in before they can be taken out. Therefore, if acquire() is called when no object is placed, it will return null. Therefore, the object pool can be encapsulated as follows to facilitate its use:

public class MyPooledClass { private static final SynchronizedPool<MyPooledClass> sPool = new SynchronizedPool<MyPooledClass>(10); public static MyPooledClass obtain() { MyPooledClass instance = sPool.acquire(); return (instance ! = null) ? instance : new MyPooledClass(); } public void recycle() { // Clear state if needed. sPool.release(this); }..}Copy the code

A demo written by myself

public class Company { private String id; private String name; private static final Pools.SynchronizedPool<Company> mPool = new Pools.SynchronizedPool<>(10); Public static Company obtain(){Company Company = mPool. Acquire (); if(company! =null){ return company; }else{ return new Company(); Public void recycle(){mPool. Release (this); }}Copy the code