安卓CPU高占用问题定位方法

今天测试报了一个问题说我们都某个应用从top命令看cpu占用到了百分之90几,这种问题要怎么分析呢?

首先adb连接上去之后用logcat | grep 应用进程pid查看看看有没有频繁打印日志,如果有的话根据日志就能比较简单的定位到代码,但这次我看了下并没有什么输出。

然后可以adb连接上去用Profiler抓一下进程的CPU使用信息,例如下图就可以看到Thread-2这个线程一直处于Running的状态没有停过,所以就是这个线程在不停占用CPU:

1.png

如果某些机器不能使用adb只能用串口查看的话还能怎么做呢?

我们可以用kill -3 应用进程pid命令强行在/data/anr下生成所有线程的堆栈文件,然后可以一个个线程分析过去看看线程运行到哪里了。当然如果线程数量比较多的时候就比较低效了,尤其是开了混淆的情况下。

但其实堆栈文件里面是有信息可以辅助我们分析是哪个线程不停在占用cpu的,例如Thread-6的线程堆栈如下:

"Thread-6" prio=5 tid=3 Native
  | group="main" sCount=1 ucsCount=0 flags=1 obj=0x75c801b8 self=0xb400007140d4ca60
  | sysTid=5030 nice=0 cgrp=default sched=0/0 handle=0x6fb6d0bcb0
  | state=R schedstat=( 43469364687935 414576037821 2915248 ) utm=2781704 stm=1565231 core=6 HZ=100
  | stack=0x6fb650c000-0x6fb650e000 stackSize=8191KB
  | held mutexes=
  native: #00 pc 00000000000a22d8  /apex/com.android.runtime/lib64/bionic/libc.so (__ioctl+8) (BuildId: 058e3ec96fa600fb840a6a6956c6b64e)
  native: #01 pc 000000000005b40c  /apex/com.android.runtime/lib64/bionic/libc.so (ioctl+156) (BuildId: 058e3ec96fa600fb840a6a6956c6b64e)
  ...

| state=R schedstat=( 43469364687935 414576037821 2915248 ) utm=2781704 stm=1565231 core=6 HZ=100这行其实能看到比较有用的信息但是可能很多人会忽略掉。

state

线程状态,root的情况下我们也可以直接cat /proc/{pid}/task/{tid}/stat获取,它和cat /proc/{pid}/stat的进程状态类似,从文档看有下面的下面的值:

  • R Running
  • S Sleeping in an interruptible wait
  • D Waiting in uninterruptible disk sleep
  • Z Zombie
  • T Stopped (on a signal) or (before Linux 2.6.33) trace stopped
  • t Tracing stop (Linux 2.6.33 onward)
  • W Paging (only before Linux 2.6.0)
  • X Dead (from Linux 2.6.0 onward)
  • x Dead (Linux 2.6.33 to 3.13 only)
  • K Wakekill (Linux 2.6.33 to 3.13 only)
  • W Waking (Linux 2.6.33 to 3.13 only)
  • P Parked (Linux 3.9 to 3.13 only)
  • I Idle (Linux 4.14 onward)

schedstat

schedstat显示了CPU的调度器统计数据,它有三个数字的含义分别是

  1. 在CPU上运行花费的时间(单位是纳秒)
  2. 在调度队列上等待的时间(单位是纳秒)
  3. CPU调度切换次数

root的情况下我们也可以直接cat /proc/{pid}/task/{tid}/schedstat获取

所以schedstat=( 43469364687935 414576037821 2915248 )表示这个线程在CPU上运行了43469364687935ns, 等待了414576037821ns, 调度切换了2915248次。

HZ

系统时间是以tick(节拍)为单位进行计算的,维基百科上是这么说的:

The system clock is typically implemented as a programmable interval timer that periodically interrupts the CPU, which then starts executing a timer interrupt service routine. This routine typically adds one tick to the system clock (a simple counter) and handles other periodic housekeeping tasks (preemption, etc.) before returning to the task the CPU was executing before the interruption.

Linux核心每隔固定周期会发出timer interrupt (IRQ 0),这个中断会触发系统时间的更新、系统资源使用率的更新、检查alarm或者delay function之类的调用,检查进程CPU的分配等。

HZ用来定义每秒有多少次timer interrupt,通过它可以计算出每个tick是多长,例如上面显示的HZ=100代表的是每个tick为1s / 100 = 10ms

utm、stm

因为系统时间以tick为单位,所以进程的用户态、内核态运行时间都是以tick为单位的。

  • utime: 用户态下使用了多少个tick的CPU
  • utime: 内核态下使用了多少个tick的CPU

由于从前面的HZ=100的到每个tick为10ms,所以utm=2781704 stm=1565231代表这个线程在用户态下使用了2781704 * 10 = 27817040ms的CPU,在内核态下使用了1565231 * 10 = 15652310ms的CPU,

它总的CPU使用总时间就是27817040ms + 15652310ms = 43469350ms和前面schedstat算出来的43469364687935ns = 43469364ms是匹配的

core

最后是CPU哪个核在执行这个线程

定位方法

有了上面这些信息之后我们就能比较快速的定位是哪个线程在不断占用cpu了:

  1. 使用kill -3 {pid}命令输出第一份堆栈
  2. 等待1分钟
  3. 使用kill -3 {pid}命令输出第二份堆栈
  4. 编写脚本对比两份堆栈里面每个线程的schedstat,看每个线程在这一分钟了使用了多久的cpu

这个脚本代码如下:

import sys
import datetime

def readTrace(path):
    thread_run_time = {}
    with open(path) as f:
        for line in f.readlines():
            if line.startswith("----- pid "):
                trace_time = datetime.datetime.strptime(line.split(" at ")[1].split(".")[0], "%Y-%m-%d %H:%M:%S")
            elif " tid=" in line:
                thread_name = line.strip().split("\"")[1]
            elif "schedstat=(" in line:
                schedstat = line.split("schedstat=( ")[1].split(")")[0].split(" ")
                thread_run_time[thread_name] = (int(schedstat[0]) + int(schedstat[1])) / 1000000000.0
    return trace_time, thread_run_time


if __name__ == '__main__':
    trace_time_start, thread_run_time_start = readTrace(sys.argv[1])
    trace_time_end, thread_run_time_end = readTrace(sys.argv[2])
    duration = (trace_time_end - trace_time_start).total_seconds()
    result = []
    for thread, thread_run_time in thread_run_time_end.items():
        if thread in thread_run_time_start:
            thread_run_in_duration = thread_run_time_end[thread] -  thread_run_time_start[thread]
            result.append([thread, thread_run_in_duration, f"{thread_run_in_duration / duration:.{2}%}"])
    for it in sorted(result, key=lambda it: it[1], reverse=True):
        print(it[0], it[1], it[2])

执行结果如下:

python3 test.py trace_04 trace_05
Thread-6 100.00285464500485 99.01%
main 0.4229402489999998 0.42%
HeapTaskDaemon 0.1186862900000012 0.12%
Signal Catcher 0.113778791 0.11%
Thread-7 0.11453095000000069 0.11%
DefaultDispatcher-worker-5 0.1086997830000005 0.11%
...

最后找到Thread-6线程在09:22:10.97708023109:23:51.419991904这100秒左右,使用了43469364687935ns - 43402540633850ns = 66824054085ns秒的cpu,然后还在调度队列上等待了414576037821ns - 381397237261ns = 33178800560ns,66824054085ns + 33178800560ns = 100002854645ns = 100.002854645s基本这100s它都在执行了:

----- pid 1834 at 2024-10-16 09:22:10.977080231+0800 -----
...
"Thread-6" prio=5 tid=3 Native
  | group="main" sCount=1 ucsCount=0 flags=1 obj=0x75c801b8 self=0xb400007140d4ca60
  | sysTid=5030 nice=0 cgrp=default sched=0/0 handle=0x6fb6d0bcb0
  | state=R schedstat=( 43402540633850 381397237261 2772868 ) utm=2777362 stm=1562891 core=5 HZ=100
  | stack=0x6fb650c000-0x6fb650e000 stackSize=8191KB
  | held mutexes=
  native: #00 pc 00000000000a22d8  /apex/com.android.runtime/lib64/bionic/libc.so (__ioctl+8) (BuildId: 058e3ec96fa600fb840a6a6956c6b64e)
  native: #01 pc 000000000005b40c  /apex/com.android.runtime/lib64/bionic/libc.so (ioctl+156) (BuildId: 058e3ec96fa600fb840a6a6956c6b64e)
  ...
----- pid 1834 at 2024-10-16 09:23:51.419991904+0800 -----
...
"Thread-6" prio=5 tid=3 Native
  | group="main" sCount=1 ucsCount=0 flags=1 obj=0x75c801b8 self=0xb400007140d4ca60
  | sysTid=5030 nice=0 cgrp=default sched=0/0 handle=0x6fb6d0bcb0
  | state=R schedstat=( 43469364687935 414576037821 2915248 ) utm=2781704 stm=1565231 core=6 HZ=100
  | stack=0x6fb650c000-0x6fb650e000 stackSize=8191KB
  | held mutexes=
  native: #00 pc 00000000000a22d8  /apex/com.android.runtime/lib64/bionic/libc.so (__ioctl+8) (BuildId: 058e3ec96fa600fb840a6a6956c6b64e)
  native: #01 pc 000000000005b40c  /apex/com.android.runtime/lib64/bionic/libc.so (ioctl+156) (BuildId: 058e3ec96fa600fb840a6a6956c6b64e)

然后就可以根据堆栈定位到底是哪里的代码逻辑异常不断占用cpu。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,386评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,939评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,851评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,953评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,971评论 5 369
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,784评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,126评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,765评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,148评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,744评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,858评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,479评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,080评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,053评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,278评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,245评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,590评论 2 343

推荐阅读更多精彩内容