Android 性能优化之卡顿优化

news2024/11/11 16:35:09

文章目录

  • Android 性能优化之卡顿优化
    • 卡顿检测
      • TraceView
        • 配置
        • 缺点
      • StricktMode
        • 配置
        • 违规代码
      • BlockCanary
        • 配置
        • 问题代码
        • 缺点
    • ANR
      • ANR原因
      • ANRWatchDog监测
      • 解决方案

Android 性能优化之卡顿优化

卡顿检测

  • TraceView
  • StricktModel
  • BlockCanary

TraceView

配置
Debug.startMethodTracing("myTrace");
Debug.stopMethodTracing();

生成的 trace 文件保存在:/storage/self/primary/Android/data/<包名>/files/myTrace.trace

缺点
  • 运行时开销大,整体变慢。
  • 容易带偏优化方向。

StricktMode

StricktMode 严苛模式,是 Android 提供的一种运行时检测机制。

策略:

  • 线程策略:
    • 自定义耗时调用:detectCustomSlowCalls
    • 磁盘读取操作:detectDiskReads
    • 网络操作:detectNetwork
  • 虚拟机策略:
    • Activity泄露:detectActivityLeaks()
    • Sqlite对象泄露:detectLeakedSqlLiteObjects
    • 检测实例数量:setClassInstanceLimit()

在 LogCat 中过滤 “StrictMode” 即可查看日志信息。

配置
public class BaseApp extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
        initStrictMode();
    }

    private void initStrictMode() {
        if (BuildConfig.DEBUG) {
            StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()
                    .detectCustomSlowCalls() //API等级11,使用StrictMode.noteSlowCode
                    .detectDiskReads()
                    .detectDiskWrites()
                    .detectNetwork()// or .detectAll() for all detectable problems
                    .penaltyLog() //在Logcat 中打印违规异常信息
                    .build());
            StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()
                    .detectActivityLeaks()
                    .detectLeakedSqlLiteObjects()
//                    .setClassInstanceLimit(Person.class, 1)
                    .detectLeakedClosableObjects() //API等级11
                    .penaltyLog()
                    .build());
        }
    }
}
违规代码

主线操作IO问题:

File externalStorage = getFilesDir();
File destFile = new File(externalStorage, "hello.txt");
try {
    OutputStream output = new FileOutputStream(destFile, true);
    output.write("I am testing io".getBytes());
    output.flush();
    output.close();
} catch (IOException e) {
    e.printStackTrace();
}

日志:

在这里插入图片描述

可以看到违规的原因和位置。

Activity内存泄露问题:

BaseApp.addActivity(this);

日志:

2024-07-15 17:50:48.075 19900-19900/com.example.anr D/StrictMode: StrictMode policy violation: android.os.strictmode.InstanceCountViolation: class com.example.anr.StuckActivity; instances=12; limit=2
        at android.os.StrictMode.setClassInstanceLimit(StrictMode.java:1)

多次进出后输出日志,存在12个实例。

BlockCanary

BlockCanary 是一个用于 Android 应用开发者的性能分析工具,它可以帮助开发者发现和解决应用中的卡顿问题。BlockCanary 通过在应用中植入检测逻辑,监控应用运行时的线程状态,并在检测到卡顿发生时记录相关信息。它的主要目的是帮助开发者追踪和分析应用中可能导致卡顿的代码段或操作。

配置

添加依赖库:

implementation 'com.github.markzhai:blockcanary-android:1.5.0'

配置:

public class BaseApplication extends Application {
    @Override
    public void onCreate() {
        BlockCanary.install(this, new AppBlockCanaryContext()).start();
    }
}
问题代码

依次执行问题代码:

private void testSleep() {
    Log.e("TAG", "sleep前");
    try {
        Thread.sleep(2000L);
    } catch (InterruptedException e) {
        throw new RuntimeException(e);
    }
    Log.e("TAG", "sleep后");
}
private void testMath() {
    Log.e("TAG", "math前");
    long start = System.currentTimeMillis();
    double result = 0;
    for (int i = 0; i < 10000000; i++) {
        result += Math.acos(Math.cos(i));
        result -= Math.asin(Math.sin(i));
        result += Math.acos(Math.cos(i));
        result -= Math.asin(Math.sin(i));
    }
    Log.e("TAG", "math" + result);
    Log.e("TAG", "耗时 " + (System.currentTimeMillis() - start));
}

BlockCanary 依次生成2个信息:

在这里插入图片描述

点击第一个进去可以看到,阻塞时间和阻塞代码:

在这里插入图片描述

点击第二个进去同样也能看到相关信息:

在这里插入图片描述

缺点
  • BlockCanary 需要在应用中植入检测逻辑,这会带来一定的性能开销。
  • 卡顿堆栈可能不准确。

ANR

ANR(Application Not Responding)是指应用程序未响应,Android 系统对于一些事件需要在一定时间范围内完成,如果超过预定时间未能得到有效响应或者响应时间过长,都会造成 ANR。

ANR原因

  • 数据导致的 ANR:频繁 GC 导致线程暂停,处理事件时间被拉长。
  • 线程阻塞或死锁导致的 ANR。
  • Binder 导致的 ANR:Binder 通信数据量过大。

在这里插入图片描述

ANRWatchDog监测

ANRWatchDog是一个用于监测Android应用程序中的ANR(应用程序无响应)的开源库。

添加依赖库:

implementation 'com.github.anrwatchdog:anrwatchdog:1.4.0'

配置:

public class BaseApp extends Application {
    @Override
    public void onCreate() {
        super.onCreate(); 
        new ANRWatchDog().start(); // 默认5000毫秒
    }
}

问题代码一:

public void onClick1(View view) {
    try {
        Thread.sleep(20000L);
    } catch (InterruptedException e) {
        throw new RuntimeException(e);
    }
}

ANRWatchDog警告:

在这里插入图片描述

问题代码二:

public void onClick2(View view) {
    SharedPreferences sp = getSharedPreferences("app", Context.MODE_PRIVATE);
    for (int i = 0; i < 100000; i++) {
        SharedPreferences.Editor edit = sp.edit();
        for (int j = 0; j < 100000; j++) {
            edit.putString("name", "aaaaa")
                    .putInt("age", 18)
                    .putBoolean("sex", true)
                    .commit();
        }
    }
}

ANRWatchDog警告:

在这里插入图片描述

解决方案

  • 将所有耗时操作如访问网络、socket 通信、查询大量 SQL 语句、复杂逻辑计算等都放在子线程中,然后通过 handler.sendMessage、runOnUIThread 等方式更新 UI。无论如何都要确保用户界面的流畅度,如果耗时操作需要让用户等待,可以在界面上显示进度条
  • 将 IO 操作放在异步线程。在一些同步的操作主线程有可能被锁,需要等待其他线程释放响应锁才能继续执行,这样会有一定的 ANR 风险,对于这种情况有时也可以用异步线程来执行相应的逻辑,另外,要避免死锁的发生
  • 使用 Thread 或 HandlerThread 时,调用 Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND)设置优先级,否则仍然会降低程序响应,因为默认 Thread 优先级和主线程相同
  • 使用 Handler 处理工作线程结果,而不是使用 Thread.wait() 或 Thread.sleep() 来阻塞主线程
  • Activity 的 onCreate() 和 onResume() 回调中避免耗时代码
  • BroadcastReceiver 中 onReceive() 代码也要尽量减少耗时,建议使用 IntentService 处理
  • 各个组件的生命周期函数都不应该有太耗时的操作,即使对于后台 Service 或 ContentProvider 来讲,虽然应用在后台运行时生命周期函数不会有用户输入引起无响应的 ANR,但其执行时间过长也会引起 Service 或 ContentProvider 的 ANR

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

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

相关文章

Python中的数据结构:五彩斑斓的糖果盒

在Python编程的世界里&#xff0c;数据结构就像是一个个五彩斑斓的糖果盒&#xff0c;每一种糖果都有其独特的味道和形状。这些多姿多彩&#xff0c;形状和味道各异的糖果盒子包括了&#xff1a;List&#xff08;列表&#xff09;、Tuple&#xff08;元组&#xff09;、Diction…

Redis主从部署

主从部署 整体架构图 需要再建两个CentOs7,过程重复单机部署 查看自己ip地址命令 ifconfig 192.168.187.137 进入redis所在目录 cd /opt/software/redis cd redis-stable 进入配置文件 vim redis.conf 修改分身1、2的配置文件 搜索replicaof replicaof 192.168.187.137 63…

笔记 2 : 课本第 3 章开始,记录 arm 的汇编指令的格式

&#xff08;13&#xff09; 介绍 arm 中的第一个汇编指令的用法 mov &#xff1a; &#xff08;14&#xff09;立即数的概念&#xff1a; &#xff08;15&#xff09; 汇编中的移位写法&#xff1a; 举例 &#xff1a; &#xff08;16&#xff09; 学习一个新的指令 cmp &a…

二叉树相关理论知识

二叉树是计算机科学中一种基础且重要的数据结构&#xff0c;它属于树形结构的一个重要类型。以下是二叉树的理论基础&#xff0c;包括定义、基本形态、特殊类型、性质以及遍历方式等方面的内容。 一、定义 二叉树&#xff08;Binary Tree&#xff09;是n&#xff08;n≥0&…

【实战系列】PostgreSQL 专栏,基于 PostgreSQL 16 版本

我的 PostgreSQL 专栏介绍及进度 20240715&#xff1a;目前整体进度已完成 85%&#xff0c;完成 16 万字&#xff0c;还有近 5 万字就截稿了。 (venv312) ➜ mypostgres git:(dev) sh scripts/word_statistics_pg_style.sh Filename …

15分钟快速了解图新地球能做什么,解决什么问题,快速入门

1.图新地球桌面端是什么 1.1官方定义 图新地球桌面端(LSV)是一款集多源数据加载、应用分析、演示汇报为一体的三维GIS 软件。采用了中科图新自主研发的国产三维地图引擎&#xff0c;支持各类无人机航测、CAD、BIM、规划成果等多源数据的加载融合;实现了BIMGIS 技术在实际业务…

所有权与生命周期:Rust 内存管理的哲学

所有权与生命周期&#xff1a;Rust内存管理的哲学 博主寄语引言&#xff1a;编程语言的内存管理困境与 Rust 的解决方案。所有权基本概念&#xff1a;资源的绝对主权生命周期的理解与应用&#xff1a;编译时的守护神借用与引用的精妙设计&#xff1a;安全与效率的和谐共舞Rust …

Golang | Leetcode Golang题解之第231题2的幂

题目&#xff1a; 题解&#xff1a; func isPowerOfTwo(n int) bool {const big 1 << 30return n > 0 && big%n 0 }

C# Opencv实现本地以图搜图

地址&#xff1a;冯腾飞/本地以图搜图

shell脚本变量和运算

一、shell变量及赋值 1.1、shell的变量 变量是用来临时保存数据的&#xff0c;并且该数据时可以变化的&#xff0c;任何一个语言都离不开变量&#xff0c;如果某个内容需要多次使用并且会重复出现&#xff0c;这样就可以使用变量了&#xff0c;如果需要修改直接修改变量就可以…

InterSystems IRIS使用python pyodbc连接 windows环境,odbc驱动安装,DSN配置,数据源配置

一、创建的数据库和数据 SELECT 1SELECT $ZVERSIONCREATE TABLE MyApp.Person ( ID INT PRIMARY KEY, Name VARCHAR(100) NOT NULL, Age INT, Gender CHAR(1) );CREATE TABLE MyApp.Person2 ( ID INT PRIMARY KEY, Name VARCHAR(100) NOT NULL, Age INT, Gender CHA…

USB 转多路串口应用软件说明

概述 PL731模块实现1个USB接口转7个异步串口(USART[1-7])功能。用于为计算机扩展异步串口&#xff0c;满足大多数一对多应用场景。支持Windows 10及以上系统&#xff0c;绝大多数Linux系统。嵌入式Linux系统有可能经过裁剪&#xff0c;需要系统开发人员确保系统安装正式的驱动…

[Unity]碰撞器的接触捕获层详解

目录 前言※关闭效果器(Effector)的遮罩接触捕获层的官方描述官方描述的翻译和注解接触捕获层作用简介接触(Contact)和捕获(Capture)配置接触捕获层的作用※接触捕获层对碰撞响应的影响需要接触捕获的物理查询需要接触捕获的物理回调注意运行时(Runtime)修改接触的相互性总结 相…

关于普通接口转AXI接口AMM Master Bridge仿真和使用

平台&#xff1a;vivado2023.1 应用场景&#xff0c;在设计的过程中&#xff0c;在xilinx内部的IP采用AXI接口协议。而我们外部的FIFO&#xff0c;BRAM等接口有时候使用的Native接口。使用AMM Master Bridge IP将普通的native接口转换为AXI接口协议。 参考文件&#xff1a;pg…

C++ | Leetcode C++题解之第229题多数元素II

题目&#xff1a; 题解&#xff1a; class Solution { public:vector<int> majorityElement(vector<int>& nums) {int n nums.size();vector<int> ans;unordered_map<int, int> cnt;for (auto & v : nums) {cnt[v];}for (auto & v : cnt…

leetcode 28.找出字符串中第一个匹配项的下标

对于Java来说直接秒呗&#xff1a; public static int strStr(String haystack, String needle) {return haystack.indexOf(needle); }

Unity UGUI Image Maskable

在Unity的UGUI系统中&#xff0c;Maskable属性用于控制UI元素是否受到父级遮罩组件的影响。以下是关于这个属性的详细说明和如何使用&#xff1a; Maskable属性 Maskable属性&#xff1a; 当你在GameObject上添加一个Image组件&#xff08;比如UI面板或按钮&#xff09;时&…

渲染100农场有哪些优势?渲染100邀请码1a12

渲染100是知名的渲染农场&#xff0c;深受广大设计师欢迎&#xff0c;比起其他农场&#xff0c;它有什么优势呢&#xff1f;我们一起来看看。 1、资源丰富 渲染100拥有强大的计算集群&#xff0c;能多线处理大规模、超复杂的场景渲染需要&#xff0c;性能卓越。2、成本低廉 渲…

【iOS】OC类与对象的本质分析

目录 前言clang常用命令对象本质探索属性的本质对象的内存大小isa 指针探究 前言 OC 代码的底层实现都是 C/C代码&#xff0c;OC 的对象都是基于 C/C 的数据结构实现的&#xff0c;实际 OC 对象的本质就是结构体&#xff0c;那到底是一个怎样的结构体呢&#xff1f; clang常用…

glibc-all-in-one+patchelf修改程序libc

主要是做堆的时候经常遇到libc小版本不对导致libcbase不对打不通的情况&#xff0c;再者&#xff0c;每个题换一个ubuntu虚拟机属实麻烦&#xff0c;所以还是回到最初也是最好用的做法&#xff1a;patch libc。 核心就是两个工具&#xff1a;glibc-all-in-one和patchlef。但是…