返回顶部
首页 > 资讯 > 移动开发 >分析Android内存泄漏的几种可能
  • 522
分享到

分析Android内存泄漏的几种可能

android内存泄漏Android 2022-06-06 08:06:18 522人浏览 薄情痞子
摘要

前言 内存泄漏简单地说就是申请了一块内存空间,使用完毕后没有释放掉。它的一般表现方式是程序运行时间越长,占用内存越多,最终用尽全部内存,整个系统崩溃。由程序申请的一块内存,且没

前言

内存泄漏简单地说就是申请了一块内存空间,使用完毕后没有释放掉。它的一般表现方式是程序运行时间越长,占用内存越多,最终用尽全部内存,整个系统崩溃。由程序申请的一块内存,且没有任何一个指针指向它,那么这块内存就泄露了。

从用户使用程序的角度来看,内存泄漏本身不会产生什么危害,作为一般的用户,根本感觉不到内存泄漏的存在。真正有危害的是内存泄漏的堆积,这会最终消耗尽系统所有的内存。从这个角度来说,一次性内存泄漏并没有什么危害,因为它不会堆积,而隐式内存泄漏危害性则非常大,因为较之于常发性和偶发性内存泄漏它更难被检测到。

Java是垃圾回收语言的一种,其优点是开发者无需特意管理内存分配,降低了应用由于局部故障(segmentation fault)导致崩溃,同时防止未释放的内存把堆栈(heap)挤爆的可能,所以写出来的代码更为安全

不幸的是,在Java中仍存在很多容易导致内存泄漏的逻辑可能(logical leak)。如果不小心,你的Android应用很容易浪费掉未释放的内存,最终导致内存用光的错误抛出

(out-of-memory,OOM)

一般内存泄漏(traditional memory leak)的原因是:当该对象的所有引用都已经释放了,对象仍未被释放。(译者注:Cursor忘记关闭等)

逻辑内存泄漏(logical memory leak)的原因是:当应用不再需要这个对象,当仍未释放该对象的所有引用。

如果持有对象的强引用,垃圾回收器是无法在内存中回收这个对象。

在Android开发中,最容易引发的内存泄漏问题的是Context。比如Activity的Context,就包含大量的内存引用,例如View Hierarchies和其他资源。一旦泄漏了Context,也意味泄漏它指向的所有对象。Android机器内存有限,太多的内存泄漏容易导致OOM。

检测逻辑内存泄漏需要主观判断,特别是对象的生命周期并不清晰。幸运的是,Activity有着明确的生命周期,很容易发现泄漏的原因。

Activity.onDestroy()
被视为Activity生命的结束,程序上来看,它应该被销毁了,或者Android系统需要回收这些内存(译者注:当内存不够时,Android会回收看不见的Activity)。

如果这个方法执行完,在堆栈中仍存在持有该Activity的强引用,垃圾回收器就无法把它标记成已回收的内存,而我们本来目的就是要回收它!

结果就是Activity存活在它的生命周期之外。

Activity是重量级对象,应该让Android系统来处理它。然而,逻辑内存泄漏总是在不经意间发生。(译者注:曾经试过一个Activity导致20M内存泄漏)。在Android中,导致潜在内存泄漏的陷阱不外乎两种:

全局进程(process-global)的static变量。这个无视应用的状态,持有Activity的强引用的怪物。

活在Activity生命周期之外的线程。没有清空对Activity的强引用。

检查一下你有没有遇到下列的情况。

Static Activities

在类中定义了静态Activity变量,把当前运行的Activity实例赋值于这个静态变量。

如果这个静态变量在Activity生命周期结束后没有清空,就导致内存泄漏。因为static变量是贯穿这个应用的生命周期的,所以被泄漏的Activity就会一直存在于应用的进程中,不会被垃圾回收器回收。


static Activity activity;
void setStaticActivity() {
   activity = this;
}
View saButton = findViewById(R.id.sa_button);
saButton.setOnClickListener(new View.OnClickListener() {
   @Override public void onClick(View v) {
     setStaticActivity();
     nextActivity();
   }
});

Memory Leak 1 – Static Activity

Static Views

类似的情况会发生在单例模式中,如果Activity经常被用到,那么在内存中保存一个实例是很实用的。正如之前所述,强制延长Activity的生命周期是相当危险而且不必要的,无论如何都不能这样做。

特殊情况:如果一个View初始化耗费大量资源,而且在一个Activity生命周期内保持不变,那可以把它变成static,加载到视图树上(View Hierachy),像这样,当Activity被销毁时,应当释放资源。(译者注:示例代码中并没有释放内存,把这个static view置null即可,但是还是不建议用这个static view的方法)


static view;
void setStaticView() {
  view = findViewById(R.id.sv_button);
}
View svButton = findViewById(R.id.sv_button);
svButton.setOnClickListener(new View.OnClickListener() {
   @Override public void onClick(View v) {
     setStaticView();
     nextActivity();
   }
});

Memory Leak 2 – Static View

Inner Classes

继续,假设Activity中有个内部类,这样做可以提高可读性和封装性。将如我们创建一个内部类,而且持有一个静态变量的引用,恭喜,内存泄漏就离你不远了(译者注:销毁的时候置空,嗯)。


private static Object inner;
void createInnerClass() {
   class InnerClass {
   }
   inner = new InnerClass();
}
View icButton = findViewById(R.id.ic_button);
icButton.setOnClickListener(new View.OnClickListener() {
   @Override public void onClick(View v) {
     createInnerClass();
     nextActivity();
   }
});

Memory Leak 3 – Inner Class

内部类的优势之一就是可以访问外部类,不幸的是,导致内存泄漏的原因,就是内部类持有外部类实例的强引用。

Anonymous Classes

相似地,匿名类也维护了外部类的引用。所以内存泄漏很容易发生,当你在Activity中定义了匿名的AsyncTsk

。当异步任务在后台执行耗时任务期间,Activity不幸被销毁了(译者注:用户退出,系统回收),这个被AsyncTask持有的Activity实例就不会被垃圾回收器回收,直到异步任务结束。


void startAsyncTask() {
   new AsyncTask<Void, Void, Void>() {
     @Override protected Void doInBackground(Void... params) {
        while(true);
     }
   }.execute();
}
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
View aicButton = findViewById(R.id.at_button);
aicButton.setOnClickListener(new View.OnClickListener() {
   @Override public void onClick(View v) {
     startAsyncTask();
     nextActivity();
   }
});


Memory Leak 4 – AsyncTask

Handler

同样道理,定义匿名的Runnable,用匿名类Handler执行。Runnable内部类会持有外部类的隐式引用,被传递到Handler的消息队列MessageQueue中,在Message消息没有被处理之前,Activity实例不会被销毁了,于是导致内存泄漏。


void createHandler() {
new Handler() {
   @Override public void handleMessage(Message message) {
     super.handleMessage(message);
   }
}.postDelayed(new Runnable() {
   @Override public void run() {
   while(true);
}
}, Long.MAX_VALUE >> 1);
}
View hButton = findViewById(R.id.h_button);
hButton.setOnClickListener(new View.OnClickListener() {
   @Override public void onClick(View v) {
      createHandler();
      nextActivity();
   }
});

Memory Leak 5 – Handler

Threads

我们再次通过Thread和TimerTask来展现内存泄漏。


void spawnThread() {
new Thread() {
   @Override public void run() {
     while(true);
   }
}.start();
}
View tButton = findViewById(R.id.t_button);
tButton.setOnClickListener(new View.OnClickListener() {
   @Override public void onClick(View v) {
     spawnThread();
     nextActivity();
   }
});


Memory Leak 6 – Thread

TimerTask

只要是匿名类的实例,不管是不是在工作线程,都会持有Activity的引用,导致内存泄漏。


oid scheduleTimer() {
new Timer().schedule(new TimerTask() {
   @Override
   public void run() {
     while(true);
   }
}, Long.MAX_VALUE >> 1);
}
View ttButton = findViewById(R.id.tt_button);
ttButton.setOnClickListener(new View.OnClickListener() {
   @Override public void onClick(View v) {
     scheduleTimer();
     nextActivity();
   }
});


Memory Leak 7 – TimerTask

Sensor Manager

最后,通过

Context.getSystemService(int name)
可以获取系统服务。这些服务工作在各自的进程中,帮助应用处理后台任务,处理硬件交互。如果需要使用这些服务,可以注册监听器,这会导致服务持有了Context的引用,如果在Activity销毁的时候没有注销这些监听器,会导致内存泄漏。


void reGISterListener() {
   SensORManager sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);
   Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ALL);
   sensorManager.registerListener(this, sensor, SensorManager.SENSOR_DELAY_FASTEST);
}
View smButton = findViewById(R.id.sm_button);
smButton.setOnClickListener(new View.OnClickListener() {
   @Override public void onClick(View v) {
     registerListener();
     nextActivity();
   }
});


Memory Leak 8 – Sensor Manager

总结

看过那么多会导致内存泄漏的例子,容易导致吃光手机的内存使垃圾回收处理更为频发,甚至最坏的情况会导致OOM。垃圾回收的操作是很昂贵的开销,会导致肉眼可见的卡顿。所以,实例化的时候注意持有的引用链,并经常进行内存泄漏检查。本文的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。希望对大家有所帮助。

您可能感兴趣的文章:Android 内存溢出和内存泄漏的问题5个Android开发中比较常见的内存泄漏问题及解决办法详解Android内存泄漏检测与MAT使用详解Android性能优化之内存泄漏Android开发:浅谈MVP模式应用与内存泄漏问题解决Android 有效的解决内存泄漏的问题实例详解Android内存泄漏实战解析Android 内存泄漏的几种可能总结谈一谈Android内存泄漏问题Android常见的几种内存泄漏小结


--结束END--

本文标题: 分析Android内存泄漏的几种可能

本文链接: https://lsjlt.com/news/24529.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 分析Android内存泄漏的几种可能
    前言 内存泄漏简单地说就是申请了一块内存空间,使用完毕后没有释放掉。它的一般表现方式是程序运行时间越长,占用内存越多,最终用尽全部内存,整个系统崩溃。由程序申请的一块内存,且没...
    99+
    2022-06-06
    android内存泄漏 Android
  • Android 内存泄漏的几种可能总结
    Java是垃圾回收语言的一种,其优点是开发者无需特意管理内存分配,降低了应用由于局部故障(segmentation fault)导致崩溃,同时防止未释放的内存把堆栈(heap)...
    99+
    2022-06-06
    内存泄漏 Android
  • Android常见的几种内存泄漏小结
    一、背景 最近在项目的版本迭代中,出现了一些内存问题的小插曲,然后自己花了一些时间优化了APP运行时内存大小的问题,特此做个总结,与大家分享。 二、简介 在Android程序开...
    99+
    2022-06-06
    小结 内存泄漏 Android
  • Android Studio 3.0上内存泄漏的示例分析
    这篇文章主要为大家展示了“Android Studio 3.0上内存泄漏的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Android Studio 3.0上内存泄漏的示例分析”这篇文章...
    99+
    2023-05-30
    android studio
  • Android 内存泄漏案例分析总结(Handler)
      在Android开发开发中,操作不当很容易引起内存泄漏,这里主要记录下平时遇到问题,包括:静态变量(也包含集合)、非静态的内部类、Handler、监听器,尤其是 Han...
    99+
    2022-06-06
    案例分析 内存泄漏 handler Android
  • 浅谈JavaScript中内存泄漏的几种情况
    目录一、内存泄漏是什么?二、垃圾回收机制1、标记清除2、引用计数三、常见的内存泄漏的情况1、意外的全局变量2、定时器造成的内存泄露3、闭包4、没有清理对DOM元素的引用5、事件监听(...
    99+
    2023-05-18
    JavaScript 内存泄漏
  • JavaScript中内存泄漏的几种情况总结
    目录1.循环引用2.定时器未清除3.DOM元素未正确删除4.全局变量未清除5.闭包未正确使用6.事件未正确解绑7.大量数据未及时清理8.使用了第三方库或框架JavaScript 中的...
    99+
    2023-05-19
    JavaScript发生内存泄漏情况 JavaScript内存泄漏解决方法 JavaScript内存泄漏
  • Android中的内存泄漏
    什么是内存泄漏 长生命周期的对象持有了短生命周期的对象,从而导致短生命周期的对象不能被释放 垃圾回收机制 垃圾回收机制分为:引用计数法、可达性分析法 引用计数法(有循环引用的问...
    99+
    2022-06-06
    内存泄漏 Android
  • Java内存泄漏的排查分析
    本篇内容介绍了“Java内存泄漏的排查分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一、由来前些日子小组内安排值班,轮流看顾我们的服务,...
    99+
    2023-06-02
  • JAVA内存泄漏的示例分析
    本篇文章给大家分享的是有关JAVA内存泄漏的示例分析,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。Java内存泄漏是每个Java程序员都会遇到的问题,程序在本地运行一切正常,可...
    99+
    2023-06-03
  • JavaScript内存泄漏实例分析
    这篇文章主要讲解了“JavaScript内存泄漏实例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“JavaScript内存泄漏实例分析”吧!js 内存泄...
    99+
    2024-04-02
  • Android性能优化之内存泄漏
      前言   对于内存泄漏,我想大家在开发中肯定都遇到过,只不过内存泄漏对我们来说并不是可见的,因为它是在堆中活动,而要想检测程序中是否有内存泄漏的产生,通常我们可以借助...
    99+
    2022-06-06
    内存泄漏 优化 Android
  • JavaScript中内存泄漏的示例分析
    这篇文章主要介绍了JavaScript中内存泄漏的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。1、识别方法chrome在performance中查看。开启开发工具P...
    99+
    2023-06-15
  • Android中常见的内存泄漏
    什么是内存泄漏当一个对象本该被回收,不需要再被使用时,有另外一个正在使用的对象持有它的引用从而导致它不能被回收,这导致本该被回收的对象不能被回收而停留在堆内存中,从而产生了内存泄漏。内存泄漏是造成应用程序OOM的主要原因之一,Android...
    99+
    2023-06-04
  • Java内存泄漏排查的示例分析
    这篇文章将为大家详细讲解有关Java内存泄漏排查的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。在一个凄凉的午夜午夜刚过,我就被一条来自监控系统的警报吵醒了。Adventory,我们的 PPC (...
    99+
    2023-06-04
  • Java内存泄漏实例排查分析
    这篇文章主要介绍“Java内存泄漏实例排查分析”,在日常操作中,相信很多人在Java内存泄漏实例排查分析问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java内存泄漏实例排查分析”的疑惑有所帮助!接下来,请跟...
    99+
    2023-06-16
  • 详解Android性能优化之内存泄漏
    综述 内存泄漏(memory leak)是指由于疏忽或错误造成程序未能释放已经不再使用的内存。那么在Android中,当一个对象持有Activity的引用,如果该对象不能被系统...
    99+
    2022-06-06
    内存泄漏 优化 Android
  • Android 内存溢出和内存泄漏的问题
    Android 内存溢出和内存泄漏的问题 在面试中,经常有面试官会问“你知道什么是内存溢出?什么是内存泄漏?怎么避免?”通过这篇文章,你可以回答出来了。 内存溢出 (OOM)是...
    99+
    2022-06-06
    内存溢出 内存泄漏 Android
  • .NET内存泄漏分析Windbg项目实例
    一:背景 讲故事 上个月有位朋友找到我,说他的程序出现了内存泄漏,不知道如何进一步分析,截图如下: 朋友这段话已经说的非常言简意赅了,那就上 windbg 说话吧。 二:Windb...
    99+
    2024-04-02
  • 剖析 JavaScript 内存泄漏的根源
    内存泄漏是指 JavaScript 对象或变量在不再需要时仍然被引用,导致应用程序的内存不断增长。这对 Web 应用程序尤其有害,因为它可能会导致性能下降,甚至崩溃。 检测内存泄漏 检测内存泄漏的第一步是使用浏览器工具(如 Chrome ...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作