Android:内存泄漏检查&内存优化

3.17Android优化

    手机移动设备的内存是有限的,需要避免内存泄漏,优化内存使用。

1.java中四种引用类型

    强引用、软引用、弱引用、虚引用。

    强引用:使用类构造方法,创建对象,当内存超出了,也不会释放对象所占内存空间;

    String str = new String(‘1223’);

    切断引用str=null;

    软引用:当内存不足时,会释放对象所占内存空间

    SoftReference softReference = new SoftReference(str);

    切断引用softReference.clear()

    弱引用:只要系统产生GC(垃圾回收),会释放对象所占空间

    WeakReference weakReference = new WeakReference(str);

    切断引用System.gc(),调用垃圾回收

    虚引用:判断对象是否已经被释放

    PhantomReference phantomReference = new PhantomReference

2.LeakCanary使用

    LeakCanary是一个内存泄漏检查的开源项目。

    网址:https://square.github.io/leakcanary/

1.添加LeakCanary到build.gradle

Android:内存泄漏检查&内存优化_第1张图片

找到build.gradle(Module:app)文件添加:

dependencies {

  // debugImplementation because LeakCanary should only run in debug builds.

  debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.2'

}

执行程序,设置过滤器Log tag:LeakCanary,可以看到:

Android:内存泄漏检查&内存优化_第2张图片

2.常见内存泄漏类型

    内存泄漏:在java运行过程中,内存泄漏是一种编程错误,使得应用程序保留不再需要的对象的引用。结果,无法回收该对象分配的内存,最终导致OutOfMemoryError。例如:一个Activity在调用其onDestroy方法后,就不再使用。但是在静态字段中保存对Activity的引用,阻碍了GC对内存的垃圾回收。

    大多数内存泄漏是由与对象生命周期相关的错误引起的。 以下是一些常见的Android错误:

    Fragment的onDestroyView方法中没有清除不再使用的View;

    Activity的Context储存为对象的字段,由于配置原因不能重新创建该对象;

    注册监听器、广播、RxJava订阅后,在其生命周期结束后忘记取消注册;

内部类导致内存泄漏

    内部类示例会隐式持有外部类引用。内部类执行线程耗时操作时,外部类Activity关闭,内部类会继续持有外部类引用,调用外部类方法,但是Activity已经无用,应该回收,这样会引起内存泄漏。

示例:

创建内部类MyThread,执行一个耗时操作,Activity点击按钮触发耗时操作,在执行过程中将退出Activity,导致内存泄漏。

public class MainActivity extends AppCompatActivity {
    private Button button1;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        button1=findViewById(R.id.btn1);
        button1.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                startTask();
            }
        });
    }

    //执行耗时操作
    public void startTask(){
        MyThread myThread=new MyThread();
        myThread.start();
    }
    private class MyThread extends  Thread{
        @Override
        public void run() {
            for(int i=0;i<100;i++){
                //耗时操作
                SystemClock.sleep(1000);
            }
        }
    }
}

Android:内存泄漏检查&内存优化_第3张图片  Android:内存泄漏检查&内存优化_第4张图片

在通知栏可以看到LeakCanary弹出通知信息,点击可以查看详细信息。

在AndroidStudio的Logcat中也可以看到类似详细信息。

处理内部类内存泄漏:将内部类改为静态内部类,这样内部类将不再隐式持有外部类的引用,当然内部类也不能访问外部类非静态字段、方法。内部类可以通过弱引用来使用外部类非静态字段、方法,这样当GC时,不会出现内存泄漏。

示例:

public class MainActivity extends AppCompatActivity {
    private Button button1;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        button1=findViewById(R.id.btn1);
        button1.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                startTask();
            }
        });
    }

    //执行耗时操作
    public void startTask(){
        MyThread myThread=new MyThread(MainActivity.this);
        myThread.start();
    }
    //修改为静态内部类,不再隐式持有外部类引用
    private static class MyThread extends  Thread{
        //通过弱引用获取Activity中非静态字段、方法
        WeakReference weakReference=null;
        public MyThread(MainActivity mainActivity){
            weakReference=new WeakReference(mainActivity);
        }
        //获取外部类对象
        public void getOutObj(){
            //获取外部类对象
            MainActivity mainActivity=weakReference.get();
            //获取成功,通过外部类对象获取属性、方法
            if (mainActivity!=null){

            }
        }
        @Override
        public void run() {
            for(int i=0;i<100;i++){
                //耗时操作
                SystemClock.sleep(1000);
            }
        }
    }
}

内部类Handler导致内存泄漏

示例:在MainActivity中创建内部类Handler,创建startMessage方法,执行延迟发送message操作,当退出Activity时,也会出现内部类的内存泄漏。解决办法也是将内部类修改为静态或者创建对应的MyHandle

你可能感兴趣的:(Android,android,LeakCanary,内存优化,内存泄漏)