喜欢我的文章,欢迎关注微信公众号「软件测试艺术」,一起学习提高。

1、 内存泄露

Android系统为每一个运行的程序都指定了一个最大运行内存,超过这个值则会触发OOM机制,反应在界面就是闪退、 Crash现象,导致OOM发生的原因比如内存泄露或者是代码不考虑后果使用大量的资源,都有可能导致OOM出现的。OOM的临界值可以通过adb shell getprop | findstr “heap”查看到:

2、 Android的GC机制

Android GC机制沿用了java的GC机制,当需要新内存去分配对象的时候而剩余不够的时候,会触发GC,把无用的对象回收掉,其中一个重要的算法便是分代式算法,这个算法把虚拟机分为年轻代、老年代和持久代,对象先分配到年轻代,然后GC多次后还存活的将会移动到老年代,老年代就不会频繁触发GC机制,一般触发频繁的都是年轻代的对象。

3、 为什么会内存泄露

上面我们知道了GC机制,那么如果GC过后程序还是没有内存,那么会发生OOM,导致GC后还是没有足够内存分配新对象的主要原因就是内存泄露了。首先要知道内存泄露也就是GC不掉的根源是生命周期长的对象持有生命周期短的对象,导致无用的对象一直无法回收。以下是几个典型的分类:

1)**静态类相关的泄露:**static对象的生命周期伴随着整个程序的生命周期,所以这块要注意不要把一些对象引用添加到static对象里面去,会造成与之关联的对象无法回收。

2)各种资源的释放:包括cursor的关闭,IO流的关闭,bitmap的回收等,进行一些带有缓存的资源一定要关闭或者释放。

3)Handler的泄露:调用handler的delay的时候,会被认为对象是有用的,导致无法回收,还有handler开启线程去下载东西没有下载完成的时候,也会因为线程导致无法回收activity;或者使用handlerThread的时候,有延迟的方法,都会导致无法回收。其主要原因在于handler是持有activity的引用,主线程不是自带一个Looper然后给handler用,导致有关联关系。

4)各种注册引用方法:比如一个常驻的后台线程处理某些时间,把当前对象注册,因为一直持有对象引用,导致这个activity一直保留,所以不用的时候需要反注册。

5)把对象缓存进容器内却忘记remove掉:有时候为了加快页面响应,结果缓存一些对象到容器内,结果越加越多,然后挂掉。

4、 系统级别的内存管理

1)LMK机制和oom_adj的值

Android内核有个专用的驱动low-memory-kill,当系统级别的内存不够的时候会根据oom_adj的值以及内存分配状况去kill掉某个进程,oom_adj可以在/proc/[pid]/oom_adj看到,并且这个值会随着进程的状态改变而改变,比如系统进程一般是-16,越大越容易被干掉。

2)5个进程的优先级

前台进程:当前运行的,基本不死 ;

可见进程:界面可以见到,比如被遮挡 ;

服务进程:进程带后台服务的,比如播放器 ;

后台进程:点击home键,但不退出,就是后台进程了,有比较大几率会被杀;

空进程:退出应用程序,还在后台保留这空进程,为的是加快启动速率,最优先。

5、 内存抖动

内存抖动是指内存频繁地分配和回收,而频繁的GC会导致卡顿,严重时还会导致OOM(主要原因还是有因为大量小的对象频繁创建,导致内存碎片,从而当需要分配内存时,虽然总体上还是有剩余内存可分配,而由于这些内存不连续,导致无法分配,系统直接就返回OOM了)

6、 内存名词VSS、RSS、PSS、USS解释

VSS - Virtual Set Size 虚拟耗用内存(包含共享库占用的内存)

RSS - Resident Set Size 实际使用物理内存(包含共享库占用的内存)

PSS - Proportional Set Size 实际使用的物理内存(比例分配共享库占用的内存)

USS - Unique Set Size 进程独自占用的物理内存(不包含共享库占用的内存)

大小规律:

一般来说内存占用大小有如下规律:VSS >= RSS >= PSS >= USS

7、 内存值获取方法

使用命令 adb shell dumpsys meminfo package_name 获取内存信息,如日历的内存信息如下:

PSS Total:进程各部分内存的消耗,是所有进程PSS相加得到系统占用内存的总和

Native Heap:Native代码分配的内存,虚拟机和Android框架分配内存。关于什么是Native代码,即非Java代码分配的内存

Dalvik Heap:Java对象分配的占据内存

Dalvik Other:类数据结构和索引占据内存

Stack:栈内存

Private Dirty:它基本上是进程内不能被分页到磁盘的内存,也不和其他进程共享,private Dirty内存是最重要的部分,因为只被自己进程使用

Private Clean:是已经映射持久文件使用的内存页(例如正在被执行的代码),因此一段时间不使用的话就可以置换出去

Heap Alloc:是Dalvik堆和本地堆分配使用的大小,它的值比Pss Total和Private Dirty大,因为进程是从Zygote中复制分裂出来的,包含了进程共享的分配部分

Ashmem:不以dalvik-开头的内存区域,匿名共享内存用来提供共享内存通过分配一个多个进程,Android匿名共享内存是基于Linux共享内存的,都是在tmpfs文件系统上新建文件,并将其映射到不同的进程空间,从而达到共享内存的目的,只是,Android在Linux的基础上进行了改造,并借助Binder+fd文件描述符实现了共享内存的传递。

Other dev:内部driver占用的内存

.so mmap:C 库代码占用的内存

.jar mmap:Java 文件代码占用的内存

.apk mmap:apk代码占用的内存

.ttf mmap:ttf 文件代码占用的内存

.dex mmap:Dex 文件代码占用的内存

Other mmap:其他文件占用的内存

8、 测试场景选择

内存出现泄漏的前提条件一定是有新的内存分配,所以测试场景会选择有新对象创建的场景,并结合用户的使用场景和频率来确定优先级。测试场景主要有以下三种情况,配合测试次数,然后可以每5次获取一次内存值进行判断,一般测试300次,如果各种内存测试完成并等待5分钟后内存没有释放,则高概率存在内存泄露:

1)新画面打开

由于新的画面打开,就会创建新的Activity和View,并有许多其他对象被创建。

测试方法:

反复进入退出需要测试的目标Activity,如果发现Activities和Views的一直在增长,则内存泄露一定发生(退出时如果手动GC,则Activities和Views的数量应该为0)

2)画面旋转

当屏幕旋转时,Orientation设置发生了改变,当前显示的Activity会被重新创建。

测试方法:进入需要测试的目标Activity,反复横竖屏切换,如果发现Activities数量等其他值一直在增长,则内存泄露一定发生

3)滑动屏幕

滑动屏幕会使屏幕中显示的对象(比如浏览器小说阅读内容)创建。

测试方法:进入需要测试的目标Activity,一直固定某个方向滑动(向左),如果发现内存值一直在增长,则内存泄露一定发生

Case例子,仅供参考:

测试过程中的值记录模板,仅供参考:

注意:

1)每个应用的脚本需要获取的信息可以直接涉及好关联应用或进程的数据值,例如测试camera时后台camera服务进程,多媒体进程、相册进程。

2)针对内存泄露的测试,需要开发自动化脚本测试,然后测试过程中获取测试的值存入execl的固定模板,测试完成后根据测试结果数据判断是否有内存泄露

9、 定位内存泄露的原因(如果是真机测试,安装一个debug版本的apk,否则monitor无法显示进程)

方法一:使用DDMS(Monitor)检测内存泄露--需要

步骤2、然后在打开DDMS, 选择Heap标签,然后点击Cause GC按钮,点击Cause GC是手动触发JAVA垃圾回收器,如下图:

如果我们要测试某个Activity是否发生内存泄露,我们可以反复进入和退出这个Activity, 再手动触发几次垃圾回收,观察上图中 data object这一栏中的 Total Size的大小是保持稳定还是有明显的变大趋势,如果有明显的变大趋势就说明这个Activity存在内存泄露的问题,需要在具体分析。

离开当前屏幕的判断方法_Android App内存泄露测试方法总结相关推荐

  1. Android App内存泄露测试方法总结

    和你一起终身学习,这里是程序员Android 经典好文推荐,通过阅读本文,您将收获以下知识点: 一.内存泄露 二. Android的GC机制 三.为什么会内存泄露 四. 系统级别的内存管理 五.内存抖 ...

  2. 离开当前屏幕的判断方法_掌中宝系列之EMG电动门调整方法

    EMG电动门调整方法 1.不带液晶屏的EMG电动门调整方法 就地操作开关按钮图 首先设定进入learn 模式(调试模式) 先按下LT按钮并保持,同时把S1拔钮开关移至Learn位置(注意顺序不能反,一 ...

  3. 离开当前屏幕的判断方法_EXCEL 一个工作表如何快速拆分多个工作表的方法

    下面说下在一个工作簿里把里面的一个工作表依据实际需求的条件内容,快速拆分成多个工作表方法. 1.打开excel文件,现在需要依据地区和国家这个条件,分别单独生成不同的工作表出来,最原始的方法是手动新建 ...

  4. java 匿名内部类内存泄露_Android 常见内存泄露 解决方案

    前言 内存泄漏(Memory Leak)是指程序中己动态分配的堆内存由于某种原因程序未释放或无法释放,造成系统内存的浪费,导致程序运行速度减慢甚至系统崩溃 (OOM) 等严重后果. 那什么情况下不能被 ...

  5. 2021-08-12 Android APP 保持屏幕常亮和取消屏幕常亮方法

    Android APP 保持屏幕常亮和取消屏幕常亮方法 一.有时候进入一个app想让屏幕常亮,这个这个可以使用 FLAG_KEEP_SCREEN_ON 功能,涉及下面函数 Window flag: a ...

  6. iphone屏幕录制_无需第三方APP,苹果iPhone手机屏幕录制的方法

    有些小伙伴在使用iPhone手机玩游戏的时候,很想把自己玩的游戏录个高清视频,再添加自己的原声解说,做个教程,分享到网上或发送给朋友,或者看到喜欢的直播需要录制下来,以后欣赏或分享给他人,但不知道怎么 ...

  7. 仿iphone顶部状态栏_无需第三方APP,苹果iPhone手机屏幕录制的方法

    有些小伙伴在使用iPhone手机玩游戏的时候,很想把自己玩的游戏录个高清视频,再添加自己的原声解说,做个教程,分享到网上或发送给朋友,或者看到喜欢的直播需要录制下来,以后欣赏或分享给他人,但不知道怎么 ...

  8. Android App定位和规避内存泄露方法研究

    from:http://site.douban.com/android/widget/notes/350758/note/167481484/ 工作中刚好用到,网上搜到的,觉得不错,与大家分享 And ...

  9. pyqt界面屏幕分辨率自适应_在Qt5和PyQt5中设置支持高分辨率屏幕自适应的方法

    在Qt5和PyQt5中设置支持高分辨率屏幕自适应的方法 PyQt5: 程序入口添加 QtCore.QCoreApplication.setAttribute(QtCore.Qt.AA_EnableHi ...

最新文章

  1. Windows锁定计算机C代码编程实现
  2. 忍不住也谈招聘应聘程序员的事
  3. SDUT-2144 图结构练习——最小生成树
  4. 程序员算事业单位吗_清北毕业出来的程序员不如三本公务员?网友:你开心就好...
  5. 重学java基础第二十课:环境配置和第一个helloWorld
  6. 为什么Unreal 4引擎能轻易实时渲染出vray要花半天才能渲染出的场景
  7. JavaScript面向对象之Function类型
  8. Linux中eclipse配置Maven,eclipse maven选项怎么配置settings
  9. HDU 1285 确定比赛名次【拓扑排序】
  10. 蒋小华老师-中高层管理专家-专职助理冰冰
  11. python之简易飞机大战
  12. 澳洲留学生面经,Java后端,阿里、头条、百度、华为、微策略
  13. 脑科学和人工智能的思考
  14. 「实在RPA学院」电商运营36计:企微批量自动加好友
  15. php怎么让浏览器崩溃,让IE6浏览器崩溃
  16. 怎么用黑白打印机打印清晰可读的PPT文档
  17. FPGA学习方向规划
  18. 2014年国内餐饮O2O大事件
  19. 银行舆情监测怎么做?
  20. PDF文件怎么转换为图片?

热门文章

  1. Qt5.15.2+VS2019安装小结
  2. Qt程序单次启动(QSingleApplication类)
  3. 【精简解释】Java中native关键字作用
  4. arch检验python_Python实现端口检测
  5. IDEA配置java《算法》第四版环境(耗时6小时,总算配置成功了,希望能给大家一点帮助)
  6. 算法练习day3——190320(对数器、归并排序)
  7. OS / linux 内核 read 操作源代码分析
  8. c# html 后台拼_c#编写html后台
  9. 信息化建设规划_苏交集团施工企业信息化建设规划分享会顺利召开
  10. java int64如何定义_java – 具有两个int属性的自定义类的hashCode是什么?