关闭 x
IT技术网
    技 采 号
    ITJS.cn - 技术改变世界
    • 实用工具
    • 菜鸟教程
    IT采购网 中国存储网 科技号 CIO智库

    IT技术网

    IT采购网
    • 首页
    • 行业资讯
    • 系统运维
      • 操作系统
        • Windows
        • Linux
        • Mac OS
      • 数据库
        • MySQL
        • Oracle
        • SQL Server
      • 网站建设
    • 人工智能
    • 半导体芯片
    • 笔记本电脑
    • 智能手机
    • 智能汽车
    • 编程语言
    IT技术网 - ITJS.CN
    首页 » 安卓开发 »Android内存优化之static使用篇

    Android内存优化之static使用篇

    2016-01-20 00:00:00 出处:segmentfault
    分享

    在Android开发中,我们经常会使用到static来修饰我们的成员变量,其本意是为了让多个对象共用一份空间,节省内存,或者是使用单例模式,让该类只生产一个实例而在整个app中使用。然而在某些时候不恰当的使用或者是编程的不规范却会造成了内存泄露现象(java上的内存泄漏指内存得不到gc的及时回收,从而造成内存占用过多的现象)

    本文中我们主要分析的是static变量对activtiy的不恰当引用而造成的内存泄漏,因为对于同一个Activity页面一般每次打开时系统都会重新生成一个该activity的对象(standard模式下),而每个activity对象一般都含有大量的视图对象和bitmap对象,假如之前的activity对象不能得到及时的回收,从而就造成了内存的泄漏现象。

    下面一边看代码一边讲解。

    单例模式不正确的获取context

    public?class?LoginManager?{
    
    ????private?Context?context;
    ????private?static?LoginManager?manager;
    
    ????public?static?LoginManager?getInstance(Context?context)?{
    ????????if?(manager?==?null)
    ????????????manager?=?new?LoginManager(context);
    ????????return?manager;
    ????}
    
    ????private?LoginManager(Context?context)?{
    ????????this.context?=?context;
    ????}

    在LoginActivity中

    public?class?LoginActivity?extends?Activity??{
    
    ????private?LoginManager?loginManager;
    ????@Override
    ????protected?void?onCreate(Bundle?savedInstanceState)?{
    ????????super.onCreate(savedInstanceState);
    ????????setContentView(R.layout.activity_login);
    ????????loginManager?=?LoginManager.getInstance(this);
    ????}

    这种方式大家应该一看就明白问题在哪里了,在LoginManager的单例中context持有了LoginActivity的this对象,即使登录成功后我们跳转到了其他Activity页面,LoginActivity的对象仍然得不到回收因为他被单例所持有,而单例的生命周期是同Application保持一致的。

    正确的获取context的方式

    public?class?LoginManager?{
    
    ????private?Context?context;
    ????private?static?LoginManager?manager;
    
    ????public?static?LoginManager?getInstance(Context?context)?{
    ????????if?(manager?==?null)
    ????????????manager?=?new?LoginManager(context);
    ????????return?manager;
    ????}
    
    ????private?LoginManager(Context?context)?{
    ????????this.context?=?context.getApplicationContext();
    ????}

    修改方式也非常简单我们单例中context不再持有Activity的context而是持有Application的context即可,因为Application本来就是单例,所以这样就不会存在内存泄漏的的现象了。

    单例模式中通过内部类持有activity对象

    第一种方式内存泄漏太过与明显,相信大家都不会犯这种错误,接下来要介绍的这种泄漏方式会比较不那么容易发现,内部类的使用造成activity对象被单例持有。

    还是看代码再分析,下面是一个单例的类:

    public?class?TestManager?{
    ????public?static?final?TestManager?INSTANCE?=?new?TestManager();
    ????private?List<MyListener>?mListenerList;
    
    ????private?TestManager()?{
    ????????mListenerList?=?new?ArrayList<MyListener>();
    ????}
    
    ????public?static?TestManager?getInstance()?{
    ????????return?INSTANCE;
    ????}
    
    ????public?void?registerListener(MyListener?listener)?{
    ????????if?(!mListenerList.contains(listener))?{
    ????????????mListenerList.add(listener);
    ????????}
    ????}
    ????public?void?unregisterListener(MyListener?listener)?{
    ????????mListenerList.remove(listener);
    ????}
    }
    
    interface?MyListener?{
    ????public?void?onSomeThingHappen();
    }

    然后是activity:

    public?class?TestActivity?extends?AppCompatActivity?{
    
    ????private?MyListener?mMyListener=new?MyListener()?{
    ????????@Override
    ????????public?void?onSomeThingHappen()?{
    ????????}
    ????};
    ????private?TestManager?testManager=TestManager.getInstance();
    ????@Override
    ????protected?void?onCreate(Bundle?savedInstanceState)?{
    ????????super.onCreate(savedInstanceState);
    ????????setContentView(R.layout.activity_test);
    ????????testManager.registerListener(mMyListener);
    ????}
    }

    我们知道在java中,非静态的内部类的对象都是会持有指向外部类对象的引用的,因此我们将内部类对象mMyListener让单例所持有时,由于mMyListener引用了我们的activity对象,因此造成activity对象也不能被回收了,从而出现内存泄漏现象。

    修改以上代码,避免内存泄漏,在activity中添加以下代码:`

    @Override
    protected?void?onDestroy()?{
    ????testManager.unregisterListener(mMyListener);
    ????super.onDestroy();
    }

    AsyncTask不正确使用造成的内存泄漏

    介绍完以上两种情况的内存泄漏后,我们在来看一种更加容易被忽略的内存泄漏现象,对于AsyncTask不正确使用造成内存泄漏的问题。

    mTask=new?AsyncTask<String,Void,Void>()
    ????????{
    @Override
    protected?Void?doInBackground(String...?params)?{
    ????????//doSamething..
    ????????return?null;
    ????????}
    ????????}.execute("a?task");

    一般我们在主线程中开启一个异步任务都是通过实现一个内部类其继承自AsyncTask类然后实现其相应的方法来完成的,那么自然的mTask就会持有对activity实例对象的引用了。查看AsyncTask的实现,我们会通过一个SerialExecutor串行线程池来对我们的任务进行排队,而这个SerialExecutor对象就是一个static final的常量。

    具体的引用关系是:

    1.我们的任务被封装在一个FutureTask的对象中(它充当一个runable的作用),FutureTask的实现也是通过内部类来实现的,因此它也为持有AsyncTask对象,而AsyncTask对象引用了activity对象,因此activity对象间接的被FutureTask对象给引用了。

    2.futuretask对象会被添加到一个ArrayDeque类型的任务队列的mTasks实例中

    3.mTasks任务队列又被SerialExecutor对象所持有,刚也说了这个SerialExecutor对象是一个static final的常量。

    具体AsyncTask的实现大家可以去参照下其源代码,我这里就通过文字描述一下其添加任务的实现过程就可以了,总之分析了这么多通过层层引用后我们的activity会被一个static变量所引用到。所以我们在使用AsyncTask的时候不宜在其中执行太耗时的操作,假设activity已经退出了,然而AsyncTask里任务还没有执行完成或者是还在排队等待执行,就会造成我们的activity对象被回收的时间延后,一段时间内内存占有率变大。

    解决方法在activity退出的时候应该调用cancel()函数

    @Override
    protected?void?onDestroy()?{
    ????//mTask.cancel(false);
    ????mTask.cancel(true);
    ????super.onDestroy();
    }

    具体cancel()里传递true or false依实际情况而定:

    1.当我们的任务还在排队没有被执行,调用cancel()无论true or false,任务会从排队队列中移除,即任务都不会被执行到了。

    2.当我们的任务已经开始执行了(doInBackground被调用),传入参数为false时并不会打断doInBackground的执行,传入参数为true时,假如我们的线程处于休眠或阻塞(如:sleep,wait)状况是会打断其执行。

    这里具体解释下cancle(true)的意义:

    mTask=new?AsyncTask<String,Void,Void>()
    ????????{
    @Override
    protected?Void?doInBackground(String...?params)?{
    ????????try?{
    ????????Thread.sleep(10000);
    ????????}?catch?(InterruptedException?e)?{
    ????????e.printStackTrace();
    ????????}
    ????????Log.d("test",?"task?is?running");
    ????????return?null;
    ????????}
    ????????}.execute("a?task");
    ????????try?{
    ????????//保证task得以执行
    ????????Thread.sleep(2000);
    ????????}?catch?(InterruptedException?e)?{
    ????????e.printStackTrace();
    ????????}
    ????????mTask.cancel(true);

    在这样的情况下我们的线程处于休眠状态调用cancel(true)方法会打断doInBackground的执行――即不会看到log语句的输出。

    但在下面的这种情况的时候却打断不了

    mTask=new?AsyncTask<String,Void,Void>()
    ????????{
    @Override
    protected?Void?doInBackground(String...?params)?{
    
    ????????Boolean?loop=true;
    ????????while?(loop)?{
    ????????Log.d("test",?"task?is?running");
    ????????}
    ????????return?null;
    ????????}
    ????????}.execute("a?task");
    ????????try?{
    ????????Thread.sleep(2000);
    ????????}?catch?(InterruptedException?e)?{
    ????????e.printStackTrace();
    ????????}
    ????????mTask.cancel(true);

    由于我们的线程不处于等待或休眠的状况及时调用cancel(true)也不能打断doInBackground的执行――现象:log函数一直在打印输出。

    解决方法:

    mTask=new?AsyncTask<String,Void,Void>()
    ????????{
    @Override
    protected?Void?doInBackground(String...?params)?{
    ????????//doSomething..
    ????????Boolean?loop=true;
    ????????while?(loop)?{
    ????????if(isCancelled())
    ????????return?null;
    ????????Log.d("test",?"task?is?running");
    ????????}
    ????????return?null;
    ????????}
    ????????}.execute("a?task");
    ????????try?{
    ????????Thread.sleep(2000);
    ????????}?catch?(InterruptedException?e)?{
    ????????e.printStackTrace();
    ????????}
    ????????mTask.cancel(true);

    这里我们通过在每次循环是检查任务是否已经被cancle掉,假如是则退出。因此对于AsyncTask我们也得注意按照正确的方式进行使用,不然也会造成程序内存泄漏的现象。

    以上内容就是在使用static时,我们需要怎么做才能优化内存的使用,当然对于以上3种情况是我们编程中使用static经常遇到的内存泄漏的情况,但仍然还有很多情况我们不易察觉到。比如:假如不做介绍,上面的第三种情况就很难察觉到,这时我们最终的内存泄漏优化方法就是:使用内存泄漏分析工具,在下一篇文章里我会参照第三种情况(AsyncTask)造成的内存泄漏,通过使用MAT工具进行分析,讲解MAT排除内存泄漏的使用方法。

    上一篇返回首页 下一篇

    声明: 此文观点不代表本站立场;转载务必保留本文链接;版权疑问请联系我们。

    别人在看

    Destoon 模板存放规则及语法参考

    Destoon系统常量与变量

    Destoon系统目录文件结构说明

    Destoon 系统安装指南

    Destoon会员公司主页模板风格添加方法

    Destoon 二次开发入门

    Microsoft 将于 2026 年 10 月终止对 Windows 11 SE 的支持

    Windows 11 存储感知如何设置?了解Windows 11 存储感知开启的好处

    Windows 11 24H2 更新灾难:系统升级了,SSD固态盘不见了...

    小米路由器买哪款?Miwifi热门路由器型号对比分析

    IT头条

    Synology 对 Office 套件进行重大 AI 更新,增强私有云的生产力和安全性

    01:43

    StorONE 的高效平台将 Storage Guardian 数据中心占用空间减少 80%

    11:03

    年赚千亿的印度能源巨头Nayara 云服务瘫痪,被微软卡了一下脖子

    12:54

    国产6nm GPU新突破!砺算科技官宣:自研TrueGPU架构7月26日发布

    01:57

    公安部:我国在售汽车搭载的“智驾”系统都不具备“自动驾驶”功能

    02:03

    技术热点

    如何删除自带的不常用应用为windows 7减负

    MySQL中多表删除方法

    改进的二值图像像素标记算法及程序实现

    windows 7 32位系统下手动修改磁盘属性例如M盘修改为F盘

    windows 7中怎么样在家庭组互传文件

    Linux应用集成MySQL数据库访问技巧

      友情链接:
    • IT采购网
    • 科技号
    • 中国存储网
    • 存储网
    • 半导体联盟
    • 医疗软件网
    • 软件中国
    • ITbrand
    • 采购中国
    • CIO智库
    • 考研题库
    • 法务网
    • AI工具网
    • 电子芯片网
    • 安全库
    • 隐私保护
    • 版权申明
    • 联系我们
    IT技术网 版权所有 © 2020-2025,京ICP备14047533号-20,Power by OK设计网

    在上方输入关键词后,回车键 开始搜索。Esc键 取消该搜索窗口。