工业设计网站排名,网站建设维护相关人员培训,厦门市建设工程在哪备案网站,网站制作 技术3.17Android优化 手机移动设备的内存是有限的,需要避免内存泄漏,优化内存使用。
1.java中四种引用类型 强引用、软引用、弱引用、虚引用。 强引用:使用类构造方法,创建对象,当内存超出了,也不会释放对象所占内存空间; String str = new String(‘1223’); 切断引用str=…3.17Android优化 手机移动设备的内存是有限的,需要避免内存泄漏,优化内存使用。
1.java中四种引用类型 强引用、软引用、弱引用、虚引用。 强引用:使用类构造方法,创建对象,当内存超出了,也不会释放对象所占内存空间; String str = new String(‘1223’); 切断引用str=null; 软引用:当内存不足时,会释放对象所占内存空间 SoftReferenceString softReference = new SoftReferenceString(str); 切断引用softReference.clear() 弱引用:只要系统产生GC(垃圾回收),会释放对象所占空间 WeakReferenceString weakReference = new WeakReferenceString(str); 切断引用System.gc(),调用垃圾回收 虚引用:判断对象是否已经被释放 PhantomReferenceString phantomReference = new PhantomReferenceString)(str); 2.LeakCanary使用 LeakCanary是一个内存泄漏检查的开源项目。 网址:https://square.github.io/leakcanary/
1.添加LeakCanary到build.gradle 找到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,可以看到: 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;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);button1=findViewById(R.id.btn1);button1.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {startTask();}});}//执行耗时操作public void startTask(){MyThread myThread=new MyThread();myThread.start();}private class MyThread extends Thread{@Overridepublic void run() {for(int i=0;i100;i++){//耗时操作SystemClock.sleep(1000);}}}
} 在通知栏可以看到LeakCanary弹出通知信息,点击可以查看详细信息。
在AndroidStudio的Logcat中也可以看到类似详细信息。
处理内部类内存泄漏:将内部类改为静态内部类,这样内部类将不再隐式持有外部类的引用,当然内部类也不能访问外部类非静态字段、方法。内部类可以通过弱引用来使用外部类非静态字段、方法,这样当GC时,不会出现内存泄漏。
示例:
public class MainActivity extends AppCompatActivity {private Button button1;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);button1=findViewById(R.id.btn1);button1.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {startTask();}});}//执行耗时操作public void startTask(){MyThread myThread=new MyThread(MainActivity.this);myThread.start();}//修改为静态内部类,不再隐式持有外部类引用private static class MyThread extends Thread{//通过弱引用获取Activity中非静态字段、方法WeakReferenceMainActivity weakReference=null;public MyThread(MainActivity mainActivity){weakReference=new WeakReferenceMainActivity(mainActivity);}//获取外部类对象public void getOutObj(){//获取外部类对象MainActivity mainActivity=weakReference.get();//获取成功,通过外部类对象获取属性、方法if (mainActivity!=null){}}@Overridepublic void run() {for(int i=0;i100;i++){//耗时操作SystemClock.sleep(1000);}}}
} 内部类Handler导致内存泄漏
示例:在MainActivity中创建内部类Handler,创建startMessage方法,执行延迟发送message操作,当退出Activity时,也会出现内部类的内存泄漏。解决办法也是将内部类修改为静态或者创建对应的MyHandle