Android性能优化第(四)篇---Allaction Tracing追踪内存分配的轨迹

2024-02-28 09:58

本文主要是介绍Android性能优化第(四)篇---Allaction Tracing追踪内存分配的轨迹,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Allaction Tracing是追踪内存分配的工具,可以很直观的看到某个操作是如何一步步分配的。

废话不多说,贴一下代码,我对这段代码进行内存分配追踪。

public class LoginActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);}public void onClick(View view) {Intent intent=new Intent(this,HomeActivity.class);startActivity(intent);}
}

public class HomeActivity extends AppCompatActivity {private ArrayList<User> mUserList = null;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_second);createObject();}private void createObject() {mUserList = new ArrayList<>();for (int i = 0; i < 10; i++) {User user = new User("wang" + i, 18 + i);mUserList.add(user);}}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20

很简单,就是用户点击登录按钮跳转到主页,我们需要做的,就是用户点击登录按钮的这个操作,内存发生了哪些变化。 

一、启动方式一,By Android Device Monitor

当我们的程序启动之后,我们首先启动Android Device Monitor,按照 选择应用进程->Start Tracking->点击登陆按钮->Get Allocations->Stop Tracking的步骤操作,会得到如下一个窗口。

内存分配图

  • 列名解释
列名解释
Alloc Order分配序列
Allocation Size分配的大小
Allocated Class被分配的对象
Thread Id线程id号
Allocated in在哪个类分配的
第二个Allocated in在哪个方法分配的

我们分析一下对象User是如何分配的,在Filter中输入User的全类名tool.test.memory.memoryleak.domain.User可以只看User的内存分配轨迹情况,如下所示。

User内存分配图
上图中可以看到,在第1337次内存分配中,分配的是User对象,占用内存32字节,处理线程Id为1,在tool.test.memory.memoryleak.HomeActivity中的createObject方法中分配的。下面这段log可以知道User这个对象时如何被创建出来的。

  at tool.test.memory.memoryleak.HomeActivity.createObject(HomeActivity.java:24)    at tool.test.memory.memoryleak.HomeActivity.onCreate(HomeActivity.java:18)    at android.app.Activity.performCreate(Activity.java:5975) at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1111)    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2417) at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2526)  at android.app.ActivityThread.access$800(ActivityThread.java:169)    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1421)  at android.os.Handler.dispatchMessage(Handler.java:111)   at android.os.Looper.loop(Looper.java:194)    at android.app.ActivityThread.main(ActivityThread.java:5549)  at java.lang.reflect.Method.invoke(Native Method) at java.lang.reflect.Method.invoke(Method.java:372)   at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:964)   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:759)   

Zygote, 意思是“受精卵”,系统中几乎所有的应用进程都是由Zygote进程孵化出来的,/system/bin/app_process 启动时创建了一个AppRuntime对象,通过AppRuntime对象的start方法,通过JNI调用创建了一个虚拟机实例,然后运行com.android.internal.os.ZygoteInit类的静态main方法,ActivityManagerService由SystemServer创建,所以ActivityManagerService驻留于SystemServer进程中,SystemServer向Zygote发送了消息(Socket),Zygote就fork创建子进程(子进程出来作为这个即将要启动的应用程序的进程),子进程调用android.app.ActivityThread的main函数,之后就是一个Activity的启动流程,在Activity创建之后,就走到了Activity的createObject。

参考:Android系统进程Zygote启动过程的源代码分析

二、启动方式二,By Android Monitor

追踪内存分配也可以由Android Monitor启动,如图所示,我们监控了19.5s到34.8s这段时间内存的变化。

在内存图中点击途中箭头的部分,启动追踪,再次点击就是停止追踪,随后自动生成一个alloc结尾的文件,这个文件就记录了这次追踪到的所有数据,然后会在右上角打开一个窗口。展示和第一种方式有点区别,各有所长,他有两种展现方式。

  • Group by Method:用方法来分类我们的内存分配,默认会以Group by Method来组织
  • Group by Allocator:用内存分配器来分类我们的内存分配

我们用 Group by Allocator的方式来查看一下。 
Paste_Image.png

可以看到我们自己包中,每一个类的内存分配次数和分配的大小。如果我们想看内存分配的实际在源码中发生的地方,可以选择需要跳转的对象,点击该按钮就能发现我们的源码。

三、统计图

如果你愿意一层一层一层的剥开我的心,你会发现 你会讶异,你是我 最压抑,最深处的秘密

1.png

虽然比较炫酷,但是个人觉得用途不是很大,没有第一种方式直观。

三、全局查看内存使用情况

一条简单的命令就OK 
adb shell dumpsys meminfo [package-name]

连命令都不想敲的人也可以,进入Android Monitor–>System information–>Memory Usage一路点过来,Android Studio就会生成一个全局的内存查看文件。 
33.png

列名解释
Naitve Heap Size从mallinfo usmblks获得,代表最大总共分配空间
Native Heap Alloc从mallinfo uorblks获得,总共分配空间
Native Heap Free从mallinfo fordblks获得,代表总共剩余空间

还有一栏时Objects,这里可以看到内存中Views,和Activity的数量,当前View的数量是32,Activity 的数量是2,当我们的应用完全退出时,View的数量和Activity的数量如下:

Paste_Image.png

所以这个也可以作为我们判断一个应用有没有发生内存泄露的一个重要手段。OK,Android性能优化第4篇到此结束。

Please accept mybest wishes for your happiness andsuccess

这篇关于Android性能优化第(四)篇---Allaction Tracing追踪内存分配的轨迹的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/755138

相关文章

Python通过模块化开发优化代码的技巧分享

《Python通过模块化开发优化代码的技巧分享》模块化开发就是把代码拆成一个个“零件”,该封装封装,该拆分拆分,下面小编就来和大家简单聊聊python如何用模块化开发进行代码优化吧... 目录什么是模块化开发如何拆分代码改进版:拆分成模块让模块更强大:使用 __init__.py你一定会遇到的问题模www.

SpringBoot首笔交易慢问题排查与优化方案

《SpringBoot首笔交易慢问题排查与优化方案》在我们的微服务项目中,遇到这样的问题:应用启动后,第一笔交易响应耗时高达4、5秒,而后续请求均能在毫秒级完成,这不仅触发监控告警,也极大影响了用户体... 目录问题背景排查步骤1. 日志分析2. 性能工具定位优化方案:提前预热各种资源1. Flowable

SpringBoot3实现Gzip压缩优化的技术指南

《SpringBoot3实现Gzip压缩优化的技术指南》随着Web应用的用户量和数据量增加,网络带宽和页面加载速度逐渐成为瓶颈,为了减少数据传输量,提高用户体验,我们可以使用Gzip压缩HTTP响应,... 目录1、简述2、配置2.1 添加依赖2.2 配置 Gzip 压缩3、服务端应用4、前端应用4.1 N

Android中Dialog的使用详解

《Android中Dialog的使用详解》Dialog(对话框)是Android中常用的UI组件,用于临时显示重要信息或获取用户输入,本文给大家介绍Android中Dialog的使用,感兴趣的朋友一起... 目录android中Dialog的使用详解1. 基本Dialog类型1.1 AlertDialog(

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

MyBatis 动态 SQL 优化之标签的实战与技巧(常见用法)

《MyBatis动态SQL优化之标签的实战与技巧(常见用法)》本文通过详细的示例和实际应用场景,介绍了如何有效利用这些标签来优化MyBatis配置,提升开发效率,确保SQL的高效执行和安全性,感... 目录动态SQL详解一、动态SQL的核心概念1.1 什么是动态SQL?1.2 动态SQL的优点1.3 动态S

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.

一文详解SpringBoot响应压缩功能的配置与优化

《一文详解SpringBoot响应压缩功能的配置与优化》SpringBoot的响应压缩功能基于智能协商机制,需同时满足很多条件,本文主要为大家详细介绍了SpringBoot响应压缩功能的配置与优化,需... 目录一、核心工作机制1.1 自动协商触发条件1.2 压缩处理流程二、配置方案详解2.1 基础YAML

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

MySQL中慢SQL优化的不同方式介绍

《MySQL中慢SQL优化的不同方式介绍》慢SQL的优化,主要从两个方面考虑,SQL语句本身的优化,以及数据库设计的优化,下面小编就来给大家介绍一下有哪些方式可以优化慢SQL吧... 目录避免不必要的列分页优化索引优化JOIN 的优化排序优化UNION 优化慢 SQL 的优化,主要从两个方面考虑,SQL 语