Android之内存泄漏与内存溢出

news2025/1/20 16:22:28

Android之内存泄漏与内存溢出

在这里插入图片描述

概览

内存泄漏(memory leak):是指程序在申请内存后,无法释放已申请的内存空间,导致系统无法及时回收内存并且分配给其他进程使用。通常少次数的内存无法及时回收并不会到程序造成什么影响,但是如果在内存本身就比较少获取多次导致内存无法正常回收时,就会导致内存不够用,最终导致内存溢出。

内存溢出 (out of memory):是指程序申请内存时,没有足够的内存供申请者使用,导致数据无法正常存储到内存中。也就是说给你个int类型的存储数据大小的空间,但是却存储一个long类型的数据,这样就会导致内存溢出。

内存溢出和内存泄露的关系以及区别

关系:内存泄露最终会导致内存溢出,由于系统中的内存是有限的,如果过度占用资源而不及时释放,最后会导致内存不足,从而无法给所需要存储的数据提供足够的内存,从而导致内存溢出。导致内存溢出也可能是由于在给数据分配大小时没有根据实际要求分配,最后导致分配的内存无法满足数据的需求,从而导致内存溢出。

区别:内存泄露是由于GC无法及时或者无法识别可以回收的数据进行及时的回收,导致内存的浪费;内存溢出是由于数据所需要的内存无法得到满足,导致数据无法正常存储到内存中。内存泄露的多次表现就是会导致内存溢出。

内存泄漏

  • 从有一组定义为gc root的根节点到目标对象的路径,称为可达性。此类对象也就是存活对象,不可达的对象就是应该被gc垃圾回收机制进行回收的对象。在当前应用的生命周期内不再使用的对象,依然被gc root引用,导致无法回收,既造成了内存泄漏。
  • 内存泄漏即 ML (Memory Leak) 指 程序在申请内存后,当该内存不需再使用;但 却无法被释放&归还给程序的现象。

内存泄漏带来的危害

  • 用户对单次的内存泄漏并没有什么感知,但当可用的空闲空间越来越少,GC就会更容易被触发,GC进行时会停止其他线程的工作,因此有可能会造成界面卡顿等情况。
  • 后续需要分配内存的时候,很容易导致内存空间不足而出现 OOM(内存溢出)。

常见的内存泄漏问题

    1. 资源性对象未关闭
      • 对于资源性对象不再使用时,应该立即调用它的close()函数,将其关闭,然后再置为null。
      • 例如:
        • 文件I/O流、数据库连接(SQLiteDatabase)、媒体资源(MediaPlayer)、网络连接(HttpURLConnection)、Cursor对象、ContentResolver、蓝牙连接(BluetoothSocket)、网络套接字(Socket)、Bitmap对象未关闭。
        • TimerTask定时任务、Timer计时器未取消。
        • WebView未销毁。
        • 线程池未关闭。
      • 以上所提及的资源应该在使用完或者在Activity页面销毁时及时关闭。
    1. 注册对象未注销各种Listener
      • 订阅者模式中,如果注册对象不再使用时,未及时注销,会导致订阅者列表中维持这对象的引用,阻止垃圾回收,导致内存泄露。
      • 例如:
        • BraodcastReceiver、EventBus未注销造成的内存泄漏,我们应该在Activity销毁时及时注销。
    1. 类的静态变量持有大数据对象
      • 尽量避免使用静态变量存储数据,特别是大数据对象,建议使用数据库存储。
    1. 单例造成的内存泄漏
      • 优先使用Application的Context,如需使用Activity的Context,可以在传入Context时使用弱引用进行封装,然后,在使用到的地方从弱引用中获取Context,如果获取不到,则直接return即可。
      •  public class Singleton {
             private static Singleton instance;
             private Context mContext;
             private Singleton(Context context){
                 this.mContext = context;
             }
        
         	/**
         		* 如果传入的context是activity,service的上下文,会导致内存泄漏
         		* 原因是我们的instance是一个static的静态对象,这个对象的生命周期和整个app的生命周期一样长
         		* 当activity销毁的时候,我们的这个instance仍然持有者这个activity的context,就会导致activity对象无法被释放回收,就导致了内存泄漏
             */
             public static Singleton getInstance(Context context){
                 if (instance == null) {
                     synchronized (Singleton.class){
                         if (instance == null){
                             instance = new Singleton(context);
                         }
                     }
                 }
                 return instance;
             }
         }
        
    1. 非静态内部类的静态实例
      • 非静态内部类持有外部类实例的引用,若非静态内部类的实例是静态的,便拥有app存活期整个生命周期,长期持有外部类的引用,阻止外部类实例被回收。
      • 使用内部类的情况十分常见,尤其是匿名内部类:一些接口的匿名实现类,都是内部类。
      • 解决方案:
        • (1)改为静态内部类,不再持有外部类实例的引用。
        • (2)避免申明非静态内部类的静态实例。
        • (3)将内部类抽取出来封装成一个单例,如果需要Context,没有特殊要求就使用Application Context;如果需要Activity Context,则使用完毕置空,或者使用弱引用。
    1. Handler造成的泄漏
      • Handler造成内存泄露的原因:非静态内部类或者匿名内部类使得Handler默认持有外部类的引用。在Activity销毁时,由于Handler可能有未执行完/正在执行的Message,导致Handler持有Activity的引用,进而导致GC无法回收Activity。
      • 解决办法:
        • 静态内部类+弱引用:
          •  private static class MyHandler extends Handler{
             	private final WeakReference<MineActivity> mMineActivityWeak;
             	public MyHandler(MineActivity mineActivity){
             		mMineActivityWeak = new WeakReference<>(mineActivity);
             	}
             	@Override
             	public void handleMessage(@NonNull Message msg) {
             		super.handleMessage(msg);
             		MineActivity mineActivity = mMineActivityWeak.get();
             		if(mineActivity != null){
             			mineActivity.number = 5;
                     }
                 }
             }
            
        • Activity销毁时,清空Handler中,未执行或正在执行的Callback以及Message。
          •   // 清空消息队列,移除对外部类的引用
             @Override
             protected void onDestroy() {
               	super.onDestroy();
               	mHandler.removeCallbacksAndMessages(null);
             }
            
      • 注:AsyncTask内部也是Handler机制,同样存在内存泄漏风险,但其一般是临时性的。
        • 解决办法:
          • ①静态内部类+弱引用。
          • ②在 Activity 或 Fragment 的 onDestroy() 方法中,手动取消 AsyncTask。
            •  @Override
               protected void onDestroy() {
                   super.onDestroy();
                   staticAsyncTask.cancel(true);
               }
              
          • ③使用 AsyncTaskLoader,它是一个专门设计用来避免内存泄露的异步任务处理器。它会在 Activity 或 Fragment 销毁时自动取消所有的任务,从而避免内存泄露。
    1. 集合类持有过多对象导致的泄漏
      • 集合类持有过多对象可能导致内存泄漏的原因是因为集合中的对象如果变成无用状态,但是由于被集合所持有,其内存不会及时被回收,导致内存泄漏。
      • 常见的容器类还有线程池、对象池、图片缓存池等。
      • 例如:
        • 如果一个 Activity 中持有了一个集合对象,如果该 Activity 被销毁了但是集合中的对象没有移除或者清空,那么这些对象将无法被回收,导致内存泄漏。
      • 解决办法:
        • 在集合中存储对象时,要注意及时移除或者清空集合中不再需要的对象,尽量减少无用对象在集合中存储的时间。最简单的方法是:清空集合对象并设置为null。
          •  @Override
             protected void onDestroy() {
                 mList.clear();
                 mList = null;
                 super.onDestroy();
             }
            
        • 使用弱引用的集合,将集合中的对象与应用程序的生命周期解耦,避免持有过多无用对象导致的内存泄漏问题。
    1. WebView导致的泄漏
      • 目前Android中WebView的实现存在很大的兼容性问题,Google支持各个ROM厂商自行定制自己的WebView实现,各个ROM间差异较大,且大多都存在内存泄露问题。除了调用其内部的clearCache()、clearHistory()、removeAllViews()、freeMemory()、destroy()和置null以外,一般比较粗暴有效的解决方法是:将包含WebView的Activity放在一个单独的进程中,不需要时将进程销毁,从而释放所有所占内存。
      • 解决办法:
        • 不在 xml 中定义 Webview ,这样会引用 Activity,而是在需要的时候在 Activity 中创建,并且使用 getApplicationgContext()。
          •  LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,ViewGroup.LayoutParams.MATCH_PARENT);
             mWebView = new WebView(getApplicationContext());
             mWebView.setLayoutParams(params);
             mLayout.addView(mWebView);
            
        • Activity 关闭时需要手动释放 Webview 内存。
          •  override fun onDestroy() {
                 // webview?.loadDataWithBaseURL(null, "", "text/html", "utf-8", null)
                 // webview?.clearView()
                 webview?.loadUrl("about:blank")
                 webview?.parent?.let {
                     (it as ViewGroup).removeView(webview)
                 }
                 webview?.stopLoading()
                 webview?.settings?.javaScriptEnabled = false
                 webview?.clearHistory()
                 webview?.clearCache(true)
                 webview?.removeAllViewsInLayout()
                 webview?.removeAllViews()
                 webview?.webViewClient = null
                 webview?.webChromeClient = null
                 webview?.destroy()
                 webview = null
                 super.onDestroy()
             }
            
    1. 使用ListView时造成的内存泄漏
      • 使用ListView时,如果不正确地使用Adapter,可能会导致内存泄漏。这是因为ListView的机制,它会在滚动过程中重用convertView,如果在convertView中持有了一些对象的引用并没有及时释放,就会导致内存泄漏。
        •  @Override
           public View getView(int position, View convertView, ViewGroup parent) {
               ViewHolder holder;
               if (convertView == null) {
                   convertView = mInflater.inflate(R.layout.list_item_layout, null);
                   holder = new ViewHolder();
                   holder.titleTextView = convertView.findViewById(R.id.title_text_view);
                   holder.contentTextView = convertView.findViewById(R.id.content_text_view);
                   convertView.setTag(holder);
               } else {
                   holder = (ViewHolder) convertView.getTag();
               }
               return convertView;
           }
          
           private static class ViewHolder {
               TextView titleTextView;
               TextView contentTextView;
           }
          
    1. 使用第三库传递context
      • 在项目中经常会使用各种三方库,有些三方库的初始化需要我们传入一个 Context 对象,尽量使用 Context.getApplicationContext,不要直接将 Activity 传递给其他组件。
      • 比如:在一些广告的SDK中,它可能要求要用Activity的Context,但是还是要尽量查阅资料或者咨询一下看能不能使用Application的Context。
    1. 静态View导致内存泄露
      • 有时,当一个Activity经常启动,但是对应的View读取非常耗时,我们可以通过静态View变量来保持对该Activity的rootView引用。这样就可以不用每次启动Activity都去读取并渲染View了。这确实是一个提高Activity启动速度的好方法!但是要注意,一旦View attach到我们的Window上,就会持有一个Context(即Activity)的引用。而我们的View有事一个静态变量,所以导致Activity不被回收。
      • 解决办法:
        • 在使用静态View时,需要确保在资源回收时,将静态View detach掉。
    1. 属性动画未及时关闭导致内存泄露
      • 在使用ValueAnimator或者ObjectAnimator时,如果没有及时做cancel取消动画,就可能造成内存泄露。 因为在cancel方法里,最后调用了endAnimation(); ,在endAnimation里,有个AnimationHandler的单例,会持有属性动画对象的引用
      • 解决办法:
        • 在在onDestory时,调用动画的cancel方法

内存泄漏检测

  • 内存泄漏检测神器LeakCanary。
    • LeakCanary 是 Square 公司的一个开源库。通过它可以在 App 运行过程中检测内存泄漏,当内存泄漏发生时会生成发生泄漏对象的引用链,并通知程序开发人员。
    • 原理:LeakCanary 是通过在 Application 的 registerActivityLifecycleCallbacks 方法实现对 Activity 销毁监听的,该方法主要用来统一管理所有 Activity 的生命周期。所有 Activity 在销毁时在其 OnDestory 方法中都会回调 ActivityLifecycleCallbacks 的 onActivityDestroyed 方法,而 LeakCanary 要做的就是在该方法中调用 RefWatcher.watch 方法实现对 Activity 进行内存泄漏监控。
    • 接入:
      • 添加依赖:
        •  dependencies {
           	// debugImplementation because LeakCanary should only run in debug builds.
              debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.7'
           }
          
    • 2.0以上版本会自己启动。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/731100.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

快讯|​2023 FOX Upfront 主打 Tubi

在每月一期的 Tubi 快讯中&#xff0c;你将全面及时地获取 Tubi 最新发展动态&#xff0c;欢迎关注【比图科技】&#xff0c;一起成长变强&#xff01; 2023 FOX Upfront 主打 Tubi 2023 年 5 月 15 日&#xff0c;FOX 在纽约曼哈顿中心举行一年一度的 FOX Upfront&#xff0c…

行业追踪,2023-07-03,汽车零部件开始调整,继续跟踪等待参与第二波行情吧

自动复盘 2023-07-03 成交额超过 100 亿 排名靠前&#xff0c;macd柱由绿转红 成交量要大于均线 有必要给每个行业加一个上级的归类&#xff0c;这样更能体现主流方向 rps 有时候比较滞后&#xff0c;但不少是欲杨先抑&#xff0c; 应该持续跟踪&#xff0c;等 macd 反转时参与…

【通览一百个大模型】FLAN(Google)

【通览一百个大模型】FLAN&#xff08;Google&#xff09; 作者&#xff1a;王嘉宁&#xff0c;本文章内容为原创&#xff0c;仓库链接&#xff1a;https://github.com/wjn1996/LLMs-NLP-Algo 订阅专栏【大模型&NLP&算法】可获得博主多年积累的全部NLP、大模型和算法干…

pytorch学习指南---安装anaconda ,pytorch

pytorch 和tenserflow并驾齐驱&#xff0c;但是&#xff0c;pytorch更好理解。所以选择学习pytorch。 安装anaconda&#xff1a; https://blog.csdn.net/fan18317517352/article/details/123035625 教程&#xff1a;bilibili up主&#xff1a;一只小土堆 构建pytorch空间 py…

密码学学习笔记(九):Public-Key Encryption - 公钥加密2

Rivest-Shamir-Adleman (RSA) - 经典非对称加密算法 如果我们知道&#x1d45d;, &#x1d45e; &#xff08;即&#x1d441; &#x1d45d;&#x1d45e;) 我们可以在mod N中进行反幂运算。 比如&#xff1a; 我们有一个单向陷门函数&#xff0c;非常适合加密。 “教科书式”…

【2023年成都七中NOIP联赛】游记

Day 1&#xff1a; 8:30&#xff1a; 睡懒觉&#xff0c;一觉睡到 8:30&#xff0c;“嘟嘟嘟”的电话声响起&#xff0c;迷迷糊糊睁开眼睛&#xff0c;快点&#xff0c;去成都七中…… 没错&#xff0c;我连今天要考试都不知道&#xff08;其实昨天老师给我发了消息的&#…

算法 | 滑动窗口算法笔记

滑动窗口&#xff1a;核心思想 核心思想&#xff1a;维护一个窗口&#xff08;又叫做子区间&#xff09;&#xff0c;通过调整窗口的起始位置&#xff08;start&#xff09;和终止位置&#xff08;end&#xff09;&#xff0c;来寻找符合特定条件的子区间。 滑动窗口算法常常…

包管理器 | 浅谈几个常用的包管理工具

目录 &#x1f5a5;️ 前言 ◼️ npm ◼️ cnpm ◼️ yarn ◼️ pnpm ◼️ Bower &#x1f5a5;️ 参考文献 &#x1f5a5;️ 参考资料 &#x1f5a5;️ 前言 如果你是前端开发者&#xff0c;或多或少都会接触到一些包管理工具&#xff0c;包管理工具是用于持续自动化…

17.OpenCV中的GFTTDetector类

文章目录 GFTTDetector功能OpenCV中GFTTDetector类reference 欢迎访问个人网络日志&#x1f339;&#x1f339;知行空间&#x1f339;&#x1f339; 这是使用imgproc.hpp中的goodFeaturesToTrack函数封装的类&#xff0c;其使用和goodFeaturesToTrack函数基本相似。 GFTTDetec…

wps如何加载mathtype和Endnote

为了支持国产化软件&#xff0c;弃用office套装&#xff0c;现在改用wps办公软件&#xff0c;但是写作科技论文的时候还是会出现很多的不方便&#xff0c;比如文献引用、公式排版编号等等。尽管wps自带了公式编辑器&#xff0c;然鹅这可太不方便了&#xff0c;因此把几个技巧总…

由于找不到libmmd.dll,无法继续执行代码有什么好的解决办法修复?

其实要解决由于找不到libmmd.dll,无法继续执行代码这个问题还是比较简单的&#xff0c;因为这类问题不外乎就是丢失了dll文件&#xff0c;我们只要下载安装回来就可以了&#xff0c;但有朋友还是会好奇&#xff0c;libmmd.dll到底是什么文件&#xff0c;那我们就来详细说说吧&a…

paddlenlp安装教程

一、安装cuda和cuDNN 1、安装paddlepaddle之前&#xff0c;要确保电脑已经安装了对应版本的cuda和cuDNN &#xff08;1&#xff09;安装CUDA 进入官网选择合适版本下载&#xff1a; 按照步骤安装完成即可。安装完成后进行验证&#xff0c;打开cmd命令窗口&#xff0c;输入nv…

Floyd算法学习笔记

Floyd算法学习笔记 前言 同步于 c n b l o g s cnblogs cnblogs 发布 如有错误&#xff0c;欢迎各位 dalao 批评指出。 前置芝士: 1.邻接矩阵&#xff08;Floyd要用邻接矩阵存图&#xff09; 2.动态规划思想&#xff08;最好学过&#xff0c;没学过也没有太大影响&#…

无线视频传输方案|远距离无人机图传应用,+28dBm大功率wifi图传模块

针对无人机远程图片、视频传输需求&#xff0c;市面上就有无线模块研发厂家推出了基于大功率图传WiFi模块SKW77的无人机远程视频传输解决方案。在无人机和地面的中继器内加入串口WiFi模块&#xff0c;手机通过与无人机和地面的中继器内置的串口WiFi模块给无人机传递控制信号&am…

深度理解 JAVA 动态代理

本文篇幅比较长&#xff0c;在确定您是否需要仔细阅读本文前&#xff0c;可以先思考一下下面几个问题&#xff1a; 动态代理是什么&#xff1f;如何实现动态代理&#xff1f;所有类都能实现动态代理吗&#xff1f;非目标方法是否会被代理&#xff1f;为什么 JDK 实现动态代理必…

平安养老险广东分公司积极开展“7·8全国保险公众宣传日”系列活动

2023年是全面贯彻落实党的二十大精神的开局之年&#xff0c;是实施“十四五”规划承上启下的关键之年。在国家金融监督管理总局指导、中国保险行业协会组织下&#xff0c;平安养老保险股份有限公司广东分公司&#xff08;以下简称“平安养老险广东分公司”&#xff09;以“78全…

ETHERNET/IP转MODBUS-RTU协议网关

远创智控YC-EIP-RTU是自主研发的一款ETHERNET/IP从站功能的通讯网关。该产品主要功能是将各种MODBUS-RTU设备接入到ETHERNET/IP网络中。 远创智控YC-EIP-RTU连接到ETHERNET/IP总线中做为从站使用&#xff0c;连接到MODBUS-RTU总线中做为主站或从站使用。 2.ETHERNET/IP转MODBU…

IDEA中配置Java反编译工具javap -c

IDEA中配置Java反编译工具javap -c 一、前置条件二、新建外部工具三、使用方式 欢迎访问我的个人博客&#xff1a;https://wk-blog.vip 一、前置条件 确保 IDEA 已经开启了编译 javac 。默认 IDEA 是开启的。 二、新建外部工具 首先进入 Settings ---> Tools ---> Ext…

自动化测试报告样式HTMLTestRunner、BeautifulReport、HTMLReport、Allure你喜欢哪个?

自动化测试报告样式HTMLTestRunner、BeautifulReport、HTMLReport、Allure你喜欢哪个&#xff1f; 1 框架设计(准备工作)1.1 简易框架图1.2 common/reportOut.py1.3 report1.4 testcase/test_baidu.py1.5 mian.py 2 HTMLTestRunner2.1 下载使用2.2 reportOut.py设计2.3 报告样式…

十一、框架与大数据模型

frameset标签 1、frameset标签 这种结构&#xff0c;基本上被淘汰 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title> </head> <frameset cols"25%,75%">&…