android开发技巧

Android 对象池实现原理和简单使用

2020-08-06  本文已影响0人  怡红快绿

了解Android 垃圾回收

判断对象是否可以被回收
垃圾收集算法
内存分配与回收策略

GC频繁原因

Memory Churn内存抖动,内存抖动是因为大量的对象被创建又在短时间内马上被释放。瞬间产生大量的对象会严重占用Young Generation的内存区域,当达到阀值,剩余空间不够的时候,就会触发GC。即使每次分配的对象占用了很少的内存,但是他们叠加在一起会增加Heap的压力,从而触发更多其他类型的GC。

对象复用

如果我们发现了大量临时对象的创建该如何处理呢?

首先确定问题发生的原因,对象频繁创建一般有以下几种可能:

例如:在onDraw中使用一些对象,由于onDraw方法会被程序频繁调用,所以我们不能在onDraw方法里面创建对象实例,我们可以考虑在onDraw方法外提前初始化这些对象。

能直接避免对象的频繁创建当然最好,但是有时候这些对象的初始化是不可避免的,那么我们就要考虑对象的复用,采用对象池来解决问题。

对象池工作过程

图解

在一个多线程的应用中,第二,第三,第四步操作都有可能发生并发操作。多线程的组件中分享对象导致了潜在的并发问题。也存在一种情况就是当所有对象都被借出时不能满足接下来的请求,对象池必须应对这些请求,不管是告诉组件已经没有对象可借还是允许组件等待直到有归还的对象。

Android官方对象池源码解析(见注释说明)

public final class Pools {

    public interface Pool<T> {

        /**
         * @return 从对象池取出对象
         */
        @Nullable
        T acquire();

        /**
         * 释放对象并放入对象池
         *  
         * @return true表示释放的对象成功放入对象池
         *
         * @throws IllegalStateException 如果对象已经存在于对象池中抛异常
         */
        boolean release(@NonNull T instance);
    }

    private Pools() {
        /* do nothing - hiding constructor */
    }

    /**
     * 对象池的非同步实现
     */
    public static class SimplePool<T> implements Pool<T> {
        private final Object[] mPool;  //对象池中真正用于存储对象的数组

        private int mPoolSize;  //对象池内的对象个数

        /**
         * Creates a new instance.
         *
         * @param maxPoolSize 对象池最大容量
         */
        public SimplePool(int maxPoolSize) {
            if (maxPoolSize <= 0) {
                throw new IllegalArgumentException("The max pool size must be > 0");
            }
            mPool = new Object[maxPoolSize];  //初始化对象数组
        }

         //从mPool数组中取出mPoolSize - 1位置上的对象
        @Override
        @SuppressWarnings("unchecked")
        public T acquire() {
            if (mPoolSize > 0) {
                final int lastPooledIndex = mPoolSize - 1;
                T instance = (T) mPool[lastPooledIndex];
                mPool[lastPooledIndex] = null;
                mPoolSize--;
                return instance;
            }
            return null;
        }

        //回收的对象放入mPool数组的mPoolSize 位置上
        @Override
        public boolean release(@NonNull 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(@NonNull 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 = new Object();  //用于同步加锁的对象

        /**
         * Creates a new instance.
         *
         * @param maxPoolSize The max pool size.
         *
         * @throws IllegalArgumentException If the max pool size is less than zero.
         */
        public SynchronizedPool(int maxPoolSize) {
            super(maxPoolSize);
        }

        @Override
        public T acquire() {
            synchronized (mLock) {
                return super.acquire();
            }
        }

        @Override
        public boolean release(@NonNull T element) {
            synchronized (mLock) {
                return super.release(element);
            }
        }
    }
}

优点

缺点

对象池的使用

在源码中摘取出来对象池的使用帮助类:

/**
 * Helper class for creating pools of objects. An example use looks like this:
 * 
 * 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);
 *     }
 *
 * }
 * /

参考

https://www.dazhuanlan.com/2019/12/24/5e02022da1e64/

上一篇 下一篇

猜你喜欢

热点阅读