Android中的anr定位指导与建议

news2025/1/19 8:20:56

1.背景

8月份安卓出现了一次直播间卡死(ANR)问题,且由于排查难度较大,持续了较长时间。本文针对如何快速定位安卓端出现ANR问题进行总结和探讨.

这里大致补充一下当时的情况,当时看到情景的是从某一个特定的场景下进入直播间后整个直播间界面立刻就卡住了,由于进入直播间后并没有做额外的操作就卡住,这给排查带来了非常大的困难.

2.相关排查方法

我们就以本次直播间卡住问题为例,逐一分析整个问题的排查方法和步骤.这里我想到的方法有两种,

第一种方法就是日志的获取,我们可以获取本地安卓系统提供的anr日志(trace文件)进行分析,也可以获取我们自己平时打点的日志进行分析.

第二种就是结合自己对代码的了解同时结合anr发生的契机,逐一对**代码进行注释,**然后排查到导致卡住代码的位置.

由于第二种方法比较基础,同时如果是多人开发,并不是所有的代码自己都能了如指掌,可能会加大排查难度和时间.因此我们主要针对第一种方法进行展开了解.

3.手机日志的获取和分析

手机日志的获取,我们可以分为两种.

第一种就是我们的logcat日志,logcat日志包括系统日志和我们自打点的log日志,logcat日志我们可以实时抓取,也可以存在本地,甚至上传到云端.结合日志的打印我们可以推测代码发生的大致位置,然后进行判断.但它具有较大的局限性.

第二种是系统的anr日志,也就是tarce.txt文件,这个日志是专门用来记录发生ANR时app各个线程的活动情况.因此只有anr发生时才会产生该文件.后面我们将重点分析它.

下面我们来分析整个ANR事件的排查流程.

3.1获取ANR日志导出方法

3.1.1 logcat日志获取

获取logcat日志很方便,我们可以直接在AndroidStudio控制台就能看到logcat日志.

如果我们没有AndroidStudio我们也可以通过adb工具来将log日志输入文本文件中,我们只需要在adb的目录下输入如下命令即可:

adb logcat -v time process thread >.\\\logcat.log

这里我们分别打印出了进程号和线程号,以方便后续排查.

3.1.2 trace.txt文件获取

trace.txt是我们分析anr发生时最重要的文件,我们想要获取它,可以采用如下方法:

  • 针对Android低版本手机(6.0以下),使用
adb pull data/anr/traces.txt

即可将anr日志,traces.txt文件导出

  • 针对Android高版本手机,会有权限的问题.但是谷歌给我们提供了令一个命令,就是bugreport,我们可以使用如下命令
adb bugreport

即可将整体日志导出到当前命令行对应目录下,它是一个压缩包的形式,我们将它解压,然后在FS/data/anr目录下就可以找到traces.txt文件了.

3.2ANR日志分析

3.2.1anr产生的原因

首先我们需要了解一下anr产生情况都有哪些,总结如下:

产生anr的情况有: (1)Activity按键或触摸事件在特定时间内无响应(5s)

(2)Service在特定时间内无法处理完成(bind,create,start,unbind等等)(前台20s,后台200s)

(3)BroadcastReceiver在特定时间内无法处理完成(前台10S,后台60s

(4)ContentProvider执行超时,20s

上面四种情况其实就是对应的安卓的四大组件的超时情况,从这里我们不难想到,安卓的anr机制其实就是安卓系统本身设计的一种保护性措施,如果四大组建响应超时了那么就会触发android not responding的回应.在开发中我们大多数遇到的情况都是第一种,在主线程做了耗时操作,导致堵塞,然后触发点击事件,无法及时响应.就比如我们8月份出现的一次直播间卡死现象,就是上面所说的第一种情况,我们在主线程做了耗时循环操作,导致所有UI刷新都在进行等待,同时我们如果触摸屏幕也会进行等待,最后会触发anr弹窗.

3.2.2ANR文件解读

3.2.2.1 logcat文件解读

我们在logcat文件中搜索"ANR in"关键字,我们将会得到发生anr时系统给出的log信息,如下图,如果搜索不到这样的信息,我们可以直接跳过这一步,因为可能是手机厂商自己定制系统的时候修改了framework层,所以我们可能看不到。

我们从logcat中一般能找到anr发生的大致位置,以及CPU的占用情况.这个可以帮助我们缩小查找的范围.

例如上面从9392行我们可以解读到anr发生在VideoLiveRoomActivity页面,也就是我们的直播间页面。它描述着下面的内容:

9392 12-25 22:49:41.424 E/ActivityManager( 1535): Reason: Input dispatching timed out (Waiting to send non-key event because the touched window has not finished processing certain input events that  9393 were delivered to it over 500.0ms ago.  Wait queue length: 8.  Wait queue head age: 5573.6ms.)

从这里我们可以看出,它还告诉我们anr发生的原因是在等待一个非关键事件的传递,这个事件可能是一个UI绘制的事件或者别的,这个事件是存在安卓的消息队列中,因此整个消息队列目前都处于阻塞状态中。从这里可以猜想到前面应该有一个耗时操作阻塞了整个队列。

同时,我们继续往下看可以看到下面列出来了不同进程对CPU的占用情况

12-25 22:49:41.424 E/ActivityManager( 1535): CPU usage from 5064ms to 15908ms later (2023-12-25 22:49:30.515 to 2023-12-25 22:49:41.359): 12-25 22:49:41.424 E/ActivityManager( 1535):   105% 26596/com.jingdong.app.mall: 103% user + 2.6% kernel / faults: 7121 minor 12-25 22:49:41.424 E/ActivityManager( 1535):   16% 835/surfaceflinger: 12% user + 4.1% kernel / faults: 529 minor 12-25 22:49:41.424 E/ActivityManager( 1535):   14% 1535/system\_server: 9.3% user + 4.7% kernel / faults: 10314 minor 2 major 12-25 22:49:41.424 E/ActivityManager( 1535):   12% 1043/media.codec: 5.7% user + 6.5% kernel / faults: 17482 minor 12-25 22:49:41.424 E/ActivityManager( 1535):   4.4% 770/android.hardware.graphics.composer@2.1-service: 2.1% user + 2.3% kernel / faults: 10 minor 12-25 22:49:41.424 E/ActivityManager( 1535):   2.9% 30680/com.tencent.mobileqq: 1.8% user + 1.1% kernel / faults: 3273 minor 11 major 12-25 22:49:41.424 E/ActivityManager( 1535):   2.6% 993/cnss\_diag: 2.2% user + 0.4% kernel 12-25 22:49:41.424 E/ActivityManager( 1535):   2.4% 570/irq/289-synapti: 0% user + 2.4% kernel

显而易见,com.jingdong.app.mall已经占用了105%,说明它以及几乎占满了CPU的负载,这个更加能确定我们代码的某一块正在执行着耗时操作,那么如何找到这个耗时操作呢?我们继续往下看。

3.2.2.2 trace.txt文件解读

我们想要快读定位到anr的位置,那么我就需要对trace文件拥有较为深入的了解.

因此trace.txt文件是我们分析anr发生的过程中最重要的文件,接下来我们就对它进行整体分析

我们先来分析一下trace.txt文件的整体结构,我们可以将文件大致概括为如下:

----- pid 1882 at 2022-10-02 15:37:56 ----- 进程1882的详细日志 ----- end 1882 ----- ----- pid 1565 at 2022-10-02 15:37:57 ----- 进程1565的详细日志 ----- end 1565 ----- ----- pid xxx at xxxx-xx-xx xx:xx:xx ----- 进程xxx的详细日志 ----- end xxx ----- ————————————————

从上面的格式我们可以看出,文件以每个进程分块,列出了每个进程下面各个线程的活动状态,同时,一般第一个进程就是主进程,所以我们大多数情况下只要分析第一个进程即可.

那么,每个进程中的所展示的内容又是什么呢?我们继续展开看一下:

----- pid 12061 at 2023-04java-12 13:48:06 ----- Cmd line: com.xxx.xxx Build fingerprint: 'TECNO/H571/TECNO-AX8:7.0/NRD90M/B-190624V299:user/release-keys' ABI: 'arm64' Build type: optimized /** =====================类加载信息-开始 class\_linker.cc中写入,记录类加载信息===================== */ Zygote loaded classes=16108 post zygote classes=9573 Dumping registered class loaders #0 dalvik.system.PathClassLoader: \[\], parent #1 #1 java.lang.BootClassLoader: \[\], no parent #2 dalvik.system.PathClassLoader: \[/data/app/~~vbVFUHP-Q5oukq\_bLiNj0w==/com.transsnet.boomplay.lite-R3xTRHDqhXkMGO0tbfFeGw==/base.apk:/data/app/~~vbVFUHP-Q5oukq\_bLiNj0w==/com.transsnet.boomplay.lite-R3xTRHDqhXkMGO0tbfFeGw==/base.apk!classes2.dex:/data/app/~~vbVFUHP-Q5oukq\_bLiNj0w==/com.transsnet.boomplay.lite-R3xTRHDqhXkMGO0tbfFeGw==/base.apk!classes3.dex\], parent #1 #3 dalvik.system.InMemoryDexClassLoader: \[/data/user/0/com.transsnet.boomplay.lite/Anonymous-DexFile@2764641574.jar\], parent #2 #4 dalvik.system.DelegateLastClassLoader: \[/data/user\_de/0/com.google.android.gms/app\_chimera/m/00000002/DynamiteLoader.apk\], parent #0 #5 dalvik.system.DelegateLastClassLoader: \[/data/user\_de/0/com.google.android.gms/app\_chimera/m/00000000/AdsDynamite.apk\], parent #2 #6 dalvik.system.InMemoryDexClassLoader: \[/data/user/0/com.transsnet.boomplay.lite/Anonymous-DexFile@3764699927.jar\], parent #2 #7 dalvik.system.InMemoryDexClassLoader: \[/data/user/0/com.transsnet.boomplay.lite/Anonymous-DexFile@1904604828.jar\], parent #2 #8 dalvik.system.PathClassLoader: \[/product/app/WebViewGoogle/WebViewGoogle.apk\], parent #1 #9 dalvik.system.PathClassLoader: \[/product/app/TrichromeLibrary/TrichromeLibrary.apk\], parent #1 #10 dalvik.system.DexClassLoader: \[/data/user/0/com.transsnet.boomplay.lite/cache/1658186039475.jar\], parent #2 #11 dalvik.system.DexClassLoader: \[/data/user/0/com.transsnet.boomplay.lite/app\_pccache/2/BDA423607FA5C8DB48E6A852040B4F20FC43D4B8/pcam.jar\], parent #2 Done dumping class loaders Classes initialized: 7610 in 6.478s /** =====================类加载信息-结束========================================================= */ /** =====================引用表-开始 intern\_table.cc中写入,记录强弱引用数量===================== */ Intern table: 52569 strong; 653 weak /** =====================引用表-结束=========================================================== */ /** =====================JNI加载信息-开始 java\_vm\_ext.cc中写入,记录加载的so===================== */ JNI: CheckJNI is on; globals=537 (plus 272 weak) Libraries: /data/app/com.xxx.xxx-2/lib/arm64/libxxx.so /data/app/com.xxx.xxx-2/lib/arm64/libmmkv.so /system/lib64/libandroid.so /system/lib64/libcompiler\_rt.so /system/lib64/libjavacrypto.so /system/lib64/libjnigraphics.so /system/lib64/libmedia\_jni.so /system/lib64/libwebviewchromium\_loader.so libjavacore.so libopenjdk.so (10) /** =====================JNI加载信息-结束======================================================  */ /** =====================java虚拟机堆状态-开始 heap.cc中写入,记录当前的内存状态===================== */ Heap: 1% free, 22MB/22MB; 75975 objects Dumping cumulative Gc timings Start Dumping histograms for 1 iterations for partial concurrent mark sweep ProcessMarkStack: Sum: 13.723ms 99% C.I. 0.101ms-13.010ms Avg: 4.574ms Max: 13.036ms MarkRootsCheckpoint: Sum: 12.257ms 99% C.I. 6.064ms-6.193ms Avg: 6.128ms Max: 6.193ms UpdateAndMarkImageModUnionTable: Sum: 9.253ms 99% C.I. 0.772us-5675us Avg: 544.294us Max: 6044us SweepMallocSpace: Sum: 5.684ms 99% C.I. 0.044ms-5.640ms Avg: 2.842ms Max: 5.640ms ScanGrayAllocSpaceObjects: Sum: 4.281ms 99% C.I. 0.002ms-4.257ms Avg: 2.140ms Max: 4.279ms UpdateAndMarkZygoteModUnionTable: Sum: 4.197ms 99% C.I. 4.197ms-4.197ms Avg: 4.197ms Max: 4.197ms AllocSpaceClearCards: Sum: 4.094ms 99% C.I. 0.001ms-3.971ms Avg: 1.023ms Max: 4.023ms MarkConcurrentRoots: Sum: 3.598ms 99% C.I. 0.565ms-3.025ms Avg: 1.799ms Max: 3.033ms (Paused)ScanGrayAllocSpaceObjects: Sum: 2.214ms 99% C.I. 0.004ms-2.210ms Avg: 1.107ms Max: 2.210ms ReMarkRoots: Sum: 1.367ms 99% C.I. 1.367ms-1.367ms Avg: 1.367ms Max: 1.367ms MarkAllocStackAsLive: Sum: 1.293ms 99% C.I. 1.293ms-1.293ms Avg: 1.293ms Max: 1.293ms ScanGrayImageSpaceObjects: Sum: 1.034ms 99% C.I. 0.283us-711.750us Avg: 60.823us Max: 722us (Paused)ScanGrayImageSpaceObjects: Sum: 738us 99% C.I. 0.265us-644.750us Avg: 43.411us Max: 686us (Paused)ProcessMarkStack: Sum: 225us 99% C.I. 225us-225us Avg: 225us Max: 225us SweepLargeObjects: Sum: 198us 99% C.I. 198us-198us Avg: 198us Max: 198us SweepSystemWeaks: Sum: 163us 99% C.I. 163us-163us Avg: 163us Max: 163us BindBitmaps: Sum: 113us 99% C.I. 113us-113us Avg: 113us Max: 113us MarkNonThreadRoots: Sum: 108us 99% C.I. 53us-55us Avg: 54us Max: 55us EnqueueFinalizerReferences: Sum: 105us 99% C.I. 105us-105us Avg: 105us Max: 105us ImageModUnionClearCards: Sum: 104us 99% C.I. 0.250us-39us Avg: 3.058us Max: 39us FinishPhase: Sum: 77us 99% C.I. 77us-77us Avg: 77us Max: 77us ProcessReferences: Sum: 59us 99% C.I. 59us-59us Avg: 59us Max: 59us ZygoteModUnionClearCards: Sum: 57us 99% C.I. 12us-45us Avg: 28.500us Max: 45us RevokeAllThreadLocalAllocationStacks: Sum: 51us 99% C.I. 51us-51us Avg: 51us Max: 51us MarkingPhase: Sum: 43us 99% C.I. 43us-43us Avg: 43us Max: 43us ReclaimPhase: Sum: 42us 99% C.I. 42us-42us Avg: 42us Max: 42us (Paused)PausePhase: Sum: 31us 99% C.I. 31us-31us Avg: 31us Max: 31us ProcessCards: Sum: 29us 99% C.I. 13us-16us Avg: 14.500us Max: 16us ScanGrayZygoteSpaceObjects: Sum: 25us 99% C.I. 25us-25us Avg: 25us Max: 25us PreCleanCards: Sum: 22us 99% C.I. 22us-22us Avg: 22us Max: 22us SwapBitmaps: Sum: 14us 99% C.I. 14us-14us Avg: 14us Max: 14us (Paused)ScanGrayZygoteSpaceObjects: Sum: 13us 99% C.I. 13us-13us Avg: 13us Max: 13us Sweep: Sum: 12us 99% C.I. 12us-12us Avg: 12us Max: 12us MarkRoots: Sum: 4us 99% C.I. 4us-4us Avg: 4us Max: 4us InitializePhase: Sum: 3us 99% C.I. 3us-3us Avg: 3us Max: 3us RecursiveMark: Sum: 2us 99% C.I. 2us-2us Avg: 2us Max: 2us SweepZygoteSpace: Sum: 1us 99% C.I. 1us-1us Avg: 1us Max: 1us FindDefaultSpaceBitmap: Sum: 0 99% C.I. 0ns-0ns Avg: 0ns Max: 0ns Done Dumping histograms partial concurrent mark sweep paused: Sum: 6.716ms 99% C.I. 6.716ms-6.716ms Avg: 6.716ms Max: 6.716ms partial concurrent mark sweep total time: 65.237ms mean time: 65.237ms partial concurrent mark sweep freed: 47501 objects with total size 5MB partial concurrent mark sweep throughput: 730785/s / 78MB/s Start Dumping histograms for 2 iterations for sticky concurrent mark sweep FreeList: Sum: 33.813ms 99% C.I. 4us-4566us Avg: 367.532us Max: 5846us MarkRootsCheckpoint: Sum: 29.286ms 99% C.I. 2.440ms-13.136ms Avg: 7.321ms Max: 13.198ms ScanGrayAllocSpaceObjects: Sum: 17.030ms 99% C.I. 0.002ms-10.912ms Avg: 2.128ms Max: 11.077ms ProcessMarkStack: Sum: 11.693ms 99% C.I. 0.001ms-6.876ms Avg: 1.461ms Max: 6.980ms SweepArray: Sum: 9.276ms 99% C.I. 2.812ms-6.464ms Avg: 4.638ms Max: 6.464ms MarkConcurrentRoots: Sum: 9.065ms 99% C.I. 0.010ms-6.326ms Avg: 2.266ms Max: 6.356ms ScanGrayImageSpaceObjects: Sum: 5.890ms 99% C.I. 0.288us-1678.999us Avg: 86.617us Max: 1864us MarkNonThreadRoots: Sum: 4.015ms 99% C.I. 0.045ms-3.775ms Avg: 1.003ms Max: 3.820ms ReMarkRoots: Sum: 2.579ms 99% C.I. 0.575ms-2.004ms Avg: 1.289ms Max: 2.004ms SweepSystemWeaks: Sum: 1.775ms 99% C.I. 315us-1460us Avg: 887.500us Max: 1460us (Paused)ScanGrayAllocSpaceObjects: Sum: 979us 99% C.I. 0.500us-883us Avg: 244.750us Max: 883us ScanGrayZygoteSpaceObjects: Sum: 951us 99% C.I. 11us-691us Avg: 237.750us Max: 692us MarkingPhase: Sum: 499us 99% C.I. 226us-273us Avg: 249.500us Max: 273us AllocSpaceClearCards: Sum: 457us 99% C.I. 0.333us-202us Avg: 57.125us Max: 202us BindBitmaps: Sum: 330us 99% C.I. 108us-222us Avg: 165us Max: 222us ImageModUnionClearCards: Sum: 258us 99% C.I. 0.253us-59us Avg: 3.794us Max: 59us ResetStack: Sum: 231us 99% C.I. 87us-144us Avg: 115.500us Max: 144us EnqueueFinalizerReferences: Sum: 171us 99% C.I. 20us-151us Avg: 85.500us Max: 151us ProcessReferences: Sum: 125us 99% C.I. 15us-110us Avg: 62.500us Max: 110us RevokeAllThreadLocalAllocationStacks: Sum: 121us 99% C.I. 37us-84us Avg: 60.500us Max: 84us ReclaimPhase: Sum: 111us 99% C.I. 26us-85us Avg: 55.500us Max: 85us (Paused)PausePhase: Sum: 109us 99% C.I. 35us-74us Avg: 54.500us Max: 74us FinishPhase: Sum: 82us 99% C.I. 32us-50us Avg: 41us Max: 50us ZygoteModUnionClearCards: Sum: 73us 99% C.I. 12us-27us Avg: 18.250us Max: 27us (Paused)ScanGrayImageSpaceObjects: Sum: 68us 99% C.I. 0.250us-21us Avg: 2us Max: 21us InitializePhase: Sum: 63us 99% C.I. 16us-47us Avg: 31.500us Max: 47us PreCleanCards: Sum: 51us 99% C.I. 23us-28us Avg: 25.500us Max: 28us (Paused)ProcessMarkStack: Sum: 45us 99% C.I. 1us-44us Avg: 22.500us Max: 44us SwapBitmaps: Sum: 29us 99% C.I. 10us-19us Avg: 14.500us Max: 19us ForwardSoftReferences: Sum: 20us 99% C.I. 3us-17us Avg: 10us Max: 17us (Paused)ScanGrayZygoteSpaceObjects: Sum: 18us 99% C.I. 7us-11us Avg: 9us Max: 11us MarkRoots: Sum: 11us 99% C.I. 5us-6us Avg: 5.500us Max: 6us UnBindBitmaps: Sum: 10us 99% C.I. 3us-7us Avg: 5us Max: 7us SwapStacks: Sum: 4us 99% C.I. 2us-2us Avg: 2us Max: 2us RecordFree: Sum: 3us 99% C.I. 1us-2us Avg: 1.500us Max: 2us FindDefaultSpaceBitmap: Sum: 2us 99% C.I. 1us-1us Avg: 1us Max: 1us PreSweepingGcVerification: Sum: 0 99% C.I. 0ns-0ns Avg: 0ns Max: 0ns Done Dumping histograms sticky concurrent mark sweep paused: Sum: 4.504ms 99% C.I. 2.035ms-2.469ms Avg: 2.252ms Max: 2.469ms sticky concurrent mark sweep total time: 129.294ms mean time: 64.647ms sticky concurrent mark sweep freed: 91166 objects with total size 6MB sticky concurrent mark sweep throughput: 706713/s / 51MB/s Total time spent in GC: 194.531ms Mean GC size throughput: 51MB/s Mean GC object throughput: 712498 objects/s /** 2.2 内存信息* start/ Total number of allocations 214578 Total bytes allocated 32MB Total bytes freed 10MB Free memory 335KB Free memory until GC 335KB Free memory until OOME 361MB Total memory 22MB Max memory 384MB /** 2.2 内存信息* end/ Zygote space size 1340KB Total mutator paused time: 11.220ms Total time waiting for GC to complete: 3.077us Total GC count: 3 Total GC time: 194.531ms Total blocking GC count: 0 Total blocking GC time: 0 Histogram of native allocation 0:1038,128:379,256:105,384:160,640:1,896:16,1152:2 bucket size 128 Histogram of native free 0:156,64:133,192:351,256:38,448:90,640:1 bucket size 64 /** =====================java虚拟机堆状态-结束====================================================== */ /** =====================OTA文件记录-开始 oat\_file\_manager.cc中写入,记录当前使用到的OTA文件===================== */ /data/user\_de/0/com.google.android.gms/app\_chimera/m/0000030c/oat/arm64/MeasurementDynamite.odex: speed /data/user/0/com.xxx.xxx/app\_optimized/audience\_network.dex: speed /system/framework/oat/arm64/com.android.location.provider.odex: speed /system/framework/oat/arm64/com.android.media.remotedisplay.odex: speed /data/app/com.google.android.gms-2/oat/arm64/base.odex: interpret-only /data/user\_de/0/com.google.android.gms/app\_chimera/m/00000307/oat/arm64/DynamiteLoader.odex: speed /data/user\_de/0/com.google.android.gms/app\_chimera/m/000002ee/oat/arm64/dl-AdsFdrDynamite.integ\_230500000000000.odex: speed /data/app/com.xxx.xxx-2/oat/arm64/base.odex: interpret-only /** =====================OTA文件记录-结束===================================================================== */ /** =====================JIT信息记录-开始 jit.cc中写入,记录JIT的信息===================== */ Current JIT code cache size: 17KB Current JIT data cache size: 20KB Current JIT capacity: 64KB Current number of JIT code cache entries: 16 Total number of JIT compilations: 16 Total number of JIT compilations for on stack replacement: 0 Total number of deoptimizations: 0 Total number of JIT code cache collections: 0 Memory used for stack maps: Avg: 675B Max: 2992B Min: 64B Memory used for compiled code: Avg: 1081B Max: 2648B Min: 176B Memory used for profiling info: Avg: 274B Max: 1040B Min: 32B Start Dumping histograms for 16 iterations for JIT timings Compiling: Sum: 512.109ms 99% C.I. 3.640ms-96.064ms Avg: 32.006ms Max: 96.425ms TrimMaps: Sum: 12.116ms 99% C.I. 41us-5108us Avg: 757.250us Max: 5295us Done Dumping histograms Memory used for compilation: Avg: 346KB Max: 959KB Min: 77KB ProfileSaver total\_bytes\_written=0 ProfileSaver total\_number\_of\_writes=0 ProfileSaver total\_number\_of\_code\_cache\_queries=0 ProfileSaver total\_number\_of\_skipped\_writes=0 ProfileSaver total\_number\_of\_failed\_writes=0 ProfileSaver total\_ms\_of\_sleep=2000 ProfileSaver total\_ms\_of\_work=0 ProfileSaver total\_number\_of\_foreign\_dex\_marks=2 ProfileSaver max\_number\_profile\_entries\_cached=19 ProfileSaver total\_number\_of\_hot\_spikes=0 ProfileSaver total\_number\_of\_wake\_ups=0 /** =====================JIT信息记录-结束================================================ */ /** =====================ART信息指标-开始 metrics\_common.cc中写入,记录ART虚拟机的一些关键指标===================== */ *** ART internal metrics ***   Metadata:     timestamp\_since\_start\_ms: 19952   Metrics:     ClassLoadingTotalTime: count = 17358     ClassVerificationTotalTime: count = 15303     ClassVerificationCount: count = 28     WorldStopTimeDuringGCAvg: count = 0     YoungGcCount: count = 0     FullGcCount: count = 0     TotalBytesAllocated: count = 2349168     TotalGcCollectionTime: count = 0     YoungGcThroughputAvg: count = 0     FullGcThroughputAvg: count = 0     YoungGcTracingThroughputAvg: count = 0     FullGcTracingThroughputAvg: count = 0     JitMethodCompileTotalTime: count = 28492     JitMethodCompileCount: count = 39     YoungGcCollectionTime: range = 0...60000, buckets: 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0     FullGcCollectionTime: range = 0...60000, buckets: 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0     YoungGcThroughput: range = 0...10000, buckets: 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0     FullGcThroughput: range = 0...10000, buckets: 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0     YoungGcTracingThroughput: range = 0...10000, buckets: 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0     FullGcTracingThroughput: range = 0...10000, buckets: 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 *** Done dumping ART internal metrics *** /** =====================ART信息指标-结束====================================================================== */ /** =====================线程信息-开始 thread\_list.cc中写入,记录当前进程的线程信息============================== */ suspend all histogram: Sum: 2.785ms 99% C.I. 4us-1964.800us Avg: 278.500us Max: 2044us DALVIK THREADS (71): "Signal Catcher" daemon prio=5 tid=3 Runnable   | group="system" sCount=0 dsCount=0 obj=0x12d555e0 self=0x7a8d897400   | sysTid=12067 nice=0 cgrp=default sched=0/0 handle=0x7a8cd04450   | state=R schedstat=( 53844613 20052004 79 ) utm=2 stm=2 core=0 HZ=100   | stack=0x7a8cc0a000-0x7a8cc0c000 stackSize=1005KB   | held mutexes= "mutator lock"(shared held)   native: #00 pc 000000000047c0fc  /system/lib64/libart.so (\_ZN3art15DumpNativeStackERNSt3\_\_113basic\_ostreamIcNS0\_11char\_traitsIcEEEEiP12BacktraceMapPKcPNS\_9ArtMethodEPv+220)   native: #01 pc 000000000047c0f8  /system/lib64/libart.so (\_ZN3art15DumpNativeStackERNSt3\_\_113basic\_ostreamIcNS0\_11char\_traitsIcEEEEiP12BacktraceMapPKcPNS\_9ArtMethodEPv+216)   native: #02 pc 000000000045027c  /system/lib64/libart.so (\_ZNK3art6Thread9DumpStackERNSt3\_\_113basic\_ostreamIcNS1\_11char\_traitsIcEEEEbP12BacktraceMap+480)   native: #03 pc 0000000000467de8  /system/lib64/libart.so (\_ZN3art14DumpCheckpoint3RunEPNS\_6ThreadE+832)   native: #04 pc 00000000004600d4  /system/lib64/libart.so (\_ZN3art10ThreadList13RunCheckpointEPNS\_7ClosureE+476)   native: #05 pc 000000000045fcc4  /system/lib64/libart.so (\_ZN3art10ThreadList4DumpERNSt3\_\_113basic\_ostreamIcNS1\_11char\_traitsIcEEEEb+848)   native: #06 pc 000000000045f930  /system/lib64/libart.so (\_ZN3art10ThreadList14DumpForSigQuitERNSt3\_\_113basic\_ostreamIcNS1\_11char\_traitsIcEEEE+804)   native: #07 pc 000000000043b8c4  /system/lib64/libart.so (\_ZN3art7Runtime14DumpForSigQuitERNSt3\_\_113basic\_ostreamIcNS1\_11char\_traitsIcEEEE+344)   native: #08 pc 0000000000441f44  /system/lib64/libart.so (\_ZN3art13SignalCatcher13HandleSigQuitEv+2096)   native: #09 pc 0000000000440aec  /system/lib64/libart.so (\_ZN3art13SignalCatcher3RunEPv+480)   native: #10 pc 000000000006a31c  /system/lib64/libc.so (\_ZL15\_\_pthread\_startPv+208)   native: #11 pc 000000000001db28  /system/lib64/libc.so (\_\_start\_thread+16)   (no managed stack frames) "main" prio=5 tid=1 Blocked   | group="main" sCount=1 dsCount=0 obj=0x75bc7c50 self=0x7a8d896a00   | sysTid=12061 nice=0 cgrp=default sched=0/0 handle=0x7a91e23a98   | state=S schedstat=( 401196933 229090000 643 ) utm=29 stm=10 core=4 HZ=100   | stack=0x7ff9366000-0x7ff9368000 stackSize=8MB   | held mutexes=   at android.app.SystemServiceRegistry$CachedServiceFetcher.getService(SystemServiceRegistry.java:929)   - waiting to lock <0x077d5864> (a java.lang.Object\[\]) held by thread 23   at android.app.SystemServiceRegistry.getSystemService(SystemServiceRegistry.java:886)   at android.app.ContextImpl.getSystemService(ContextImpl.java:1524)   at android.content.ContextWrapper.getSystemService(ContextWrapper.java:659)   at android.view.TouchScreenHelper.updateDisplayInfo(TouchScreenHelper.java:323)   at android.view.TouchScreenHelper.init(TouchScreenHelper.java:149)   at android.view.TouchScreenHelper.<init>(TouchScreenHelper.java:87)   at android.view.WindowManagerGlobal.getTouchScreenHelper(WindowManagerGlobal.java:173)   - locked <0x02a54acd> (a java.lang.Class<android.view.WindowManagerGlobal>)   at com.android.internal.policy.DecorView.<init>(DecorView.java:290)   at com.android.internal.policy.PhoneWindow.generateDecor(PhoneWindow.java:2376)   at com.android.internal.policy.PhoneWindow.installDecor(PhoneWindow.java:2730)   at com.android.internal.policy.PhoneWindow.getDecorView(PhoneWindow.java:2085)   at com.xxx.common.base.XxxBaseActivity.toggleTranslucent(XxxBaseActivity.java:37)   at com.xxx.common.base.XxxBaseActivity.onCreate(XxxBaseActivity.java:20)   at com.xxx.xxx.guide.ControllerActivity.onCreate(ControllerActivity.java:174)   at android.app.Activity.performCreate(Activity.java:6734)   at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1118)   at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2681)   at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2793)   at android.app.ActivityThread.-wrap12(ActivityThread.java:-1)   at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1527)   at android.os.Handler.dispatchMessage(Handler.java:110)   at android.os.Looper.loop(Looper.java:203)   at android.app.ActivityThread.main(ActivityThread.java:6301)   at java.lang.reflect.Method.invoke!(Native method)   at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1094)   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:955) "RxCachedThreadScheduler-1" daemon prio=5 tid=23 Blocked   | group="main" sCount=1 dsCount=0 obj=0x12f61af0 self=0x7a8211fa00   | sysTid=12096 nice=0 cgrp=default sched=0/0 handle=0x7a68c66450   | state=S schedstat=( 164816616 182495075 396 ) utm=13 stm=2 core=7 HZ=100   | stack=0x7a68b64000-0x7a68b66000 stackSize=1037KB   | held mutexes=   at android.view.WindowManagerGlobal.getInstance(WindowManagerGlobal.java:182)   - waiting to lock <0x02a54acd> (a java.lang.Class<android.view.WindowManagerGlobal>) held by thread 1   at android.view.WindowManagerImpl.<init>(WindowManagerImpl.java:58)   at android.view.WindowManagerImpl.<init>(WindowManagerImpl.java:65)   at android.app.SystemServiceRegistry$50.createService(SystemServiceRegistry.java:599)   at android.app.SystemServiceRegistry$50.createService(SystemServiceRegistry.java:598)   at android.app.SystemServiceRegistry$CachedServiceFetcher.getService(SystemServiceRegistry.java:933)   - locked <0x077d5864> (a java.lang.Object\[\])   at android.app.SystemServiceRegistry.getSystemService(SystemServiceRegistry.java:886)   at android.app.ContextImpl.getSystemService(ContextImpl.java:1524)   at android.content.ContextWrapper.getSystemService(ContextWrapper.java:659)   at me.jessyan.autosize.utils.ScreenUtils.getScreenSize(ScreenUtils.java:62)   at me.jessyan.autosize.AutoSizeConfig.init(AutoSizeConfig.java:247)   at me.jessyan.autosize.AutoSizeConfig.init(AutoSizeConfig.java:205)   at me.jessyan.autosize.AutoSize.checkAndInit(AutoSize.java:84)   at com.xxx.common.base.XxxApplicationInitor.initAutoSize(MusicApplicationInitor.java:404)   at com.xxx.common.base.XxxApplicationInitor.thirdSDKInit(MusicApplicationInitor.java:414)   at com.xxx.common.base.XxxApplicationInitor.preInit(MusicApplicationInitor.java:438)   at com.xxx.common.base.XxxApplicationInitor.access$800(MusicApplicationInitor.java:109)   at com.xxx.common.base.XxxApplicationInitor$10.subscribe(MusicApplicationInitor.java:231)   at io.reactivex.internal.operators.observable.ObservableCreate.subscribeActual(ObservableCreate.java:40)   at io.reactivex.Observable.subscribe(Observable.java:12284)   at io.reactivex.internal.operators.observable.ObservableSubscribeOn$SubscribeTask.run(ObservableSubscribeOn.java:96)   at io.reactivex.Scheduler$DisposeTask.run(Scheduler.java:578)   at io.reactivex.internal.schedulers.ScheduledRunnable.run(ScheduledRunnable.java:66)   at io.reactivex.internal.schedulers.ScheduledRunnable.call(ScheduledRunnable.java:57)   at java.util.concurrent.FutureTask.run(FutureTask.java:237)   at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.run(ScheduledThreadPoolExecutor.java:272)   at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1133)   at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:607)   at java.lang.Thread.run(Thread.java:761) "Binder:1002\_1" prio=5 tid=7 Native   | group="main" sCount=1 dsCount=0 obj=0x12c01940 self=0x7a8d8c3400   | sysTid=1020 nice=0 cgrp=default sched=0/0 handle=0x7a869be450   | state=S schedstat=( 19616954905 44253949137 66794 ) utm=1314 stm=647 core=1 HZ=100   | stack=0x7a868c4000-0x7a868c6000 stackSize=1005KB   | held mutexes=   kernel: (couldn't read /proc/self/task/1020/stack)   native: #00 pc 000000000006c840  /system/lib64/libc.so (\_\_ioctl+4)   native: #01 pc 000000000001fb60  /system/lib64/libc.so (ioctl+140)   native: #02 pc 0000000000055698  /system/lib64/libbinder.so (\_ZN7android14IPCThreadState14talkWithDriverEb+264)   native: #03 pc 00000000000557fc  /system/lib64/libbinder.so (\_ZN7android14IPCThreadState20getAndExecuteCommandEv+24)   native: #04 pc 0000000000055f4c  /system/lib64/libbinder.so (\_ZN7android14IPCThreadState14joinThreadPoolEb+72)   native: #05 pc 00000000000739f8  /system/lib64/libbinder.so (???)   native: #06 pc 0000000000012598  /system/lib64/libutils.so (\_ZN7android6Thread11\_threadLoopEPv+272)   native: #07 pc 00000000000a2618  /system/lib64/libandroid\_runtime.so (\_ZN7android14AndroidRuntime15javaThreadShellEPv+128)   native: #08 pc 000000000006a31c  /system/lib64/libc.so (\_ZL15\_\_pthread\_startPv+208)   native: #09 pc 000000000001db28  /system/lib64/libc.so (\_\_start_thread+16)   (no managed stack frames)      ...   /** =====================线程信息-结束============================================================================*/

从上面文件中,我们能看到了记录一个进程中活动信息的所有内容,并且内容十分丰富繁多,原文件更大,我做了部分删减,这样能更加清晰的看出整个进程的所展示信息的框架,其中最重要的就是线程的活动状态,以及每个线程的堆栈信息.

下面我们对我们的主线程,也就是main线程的具体参数,作出解释.

具体解释如下:

所在行内容解释
1“main”线程名
1prio=5线程优先级
1tid=1线程标识符
1Blocked线程状态,表明当前线程处于阻塞状态,其它状态还有,waiting:线程等待状态,也就是挂起状态runnalbe:线程运行状态native:native是native中的线程状态,表明正在实行jni本地函数,对应java中runnable
2group=“main”线程所属的线程组
2sCount=1程被正常挂起的次数
2dsCount=0线程因调试而挂起次数
2flags=1
2obj=0x75c4db98线程的Java对象的地址
2self=0x7b08e14c00线程本身的地址
3sysTid=2468Linux下的内核线程id
3nice=0线程的调度优先级
3cgrp=default调度属组
3sched=0/0线程的调度策略和优先级
3handle=0x7b8e933548线程的处理函数地址
4state=S调度状态
4schedstat=( 562066147 3061697256 542 )执行时间记录:三个值分别表示线程在cpu上执行的时间、线程的等待时间和线程执行的时间片长度
4utm=31用户态CPU占用次数,由于频率是100,则占用31次就等于31*10ms=310ms
4stm=24内核态CPU占用次数
4core=2最后执行这个线程的CPU核序号
4HZ=100运行频率,一秒100次
5stack=0x7fe841d000-0x7fe841f000虚拟机栈地址
5stackSize=8MB虚拟机栈大小
6at android.os.BinderProxy.transactNative(Native method) at android.os.BinderProxy.transact(Binder.java:1127)此时采集到的APP对应线程的对应状态,也就是当时代码所执行的位置,这个信息对我们来说尤其重要,是我们分析anr的关键依据.

上表中以主线程为例,对线程的每个参数都做出了解释,下面的子线程也同样适用.读懂了每个线程中各个参数的含义,那么最需要关心的是什么呢?

答案就是线程的状态,以及线程下打印出来对应的堆栈代码信息.线程状态可以帮助我们判断anr发生在线程层面的原因,堆栈信息可以帮我我们分析anr发生的具体代码位置.

其中线程常见状态解释如下:

  • Blocked状态

也就是阻塞状态,一般主线程发生blocked,说明它在等待获取监视器锁(synchronized 锁)时,如果锁被其他线程占用,就会进入Blocked状态。示例如下:

"main" tid=1 systid=17151 blocked | waiting to lock <0x05dfb3bd> (java.lang.Object) held by thread 46        at android.app.ContextImpl.getExternalCacheDirs(ContextImpl.java:836)        at android.content.ContextWrapper.getExternalCacheDirs(ContextWrapper.java:311)        at androidx.core.content.ContextCompat$Api19Impl.getExternalCacheDirs(ContextCompat.java:840)        at androidx.core.content.ContextCompat.getExternalCacheDirs(ContextCompat.java:459)        at androidx.core.content.FileProvider.parsePathStrategy(FileProvider.java:696)        at androidx.core.content.FileProvider.getPathStrategy(FileProvider.java:635)        at androidx.core.content.FileProvider.attachInfo(FileProvider.java:416)        at android.app.ActivityThread.installProvider(ActivityThread.java:7673)        at android.app.ActivityThread.installContentProviders(ActivityThread.java:7209)        at android.app.ActivityThread.handleBindApplication(ActivityThread.java:7121)        at android.app.ActivityThread.access$1600(ActivityThread.java:268)        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2056)        at android.os.Handler.dispatchMessage(Handler.java:106)        at android.os.Looper.loop(Looper.java:268)        at android.app.ActivityThread.main(ActivityThread.java:8107)        at java.lang.reflect.Method.invoke(Method.java)        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:627)        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:997)
  • Waiting状态

线程在等待其他线程显式地通知或中断时,会进入Waiting状态。

当线程调用Object.wait()Thread.join()或类似的方法时,线程会进入Waiting状态,等待其他线程的通知。示例如下:

"main" prio=5 tid=1 Waiting   | group="main" sCount=1 dsCount=0 flags=1 obj=0x70df6868 self=0xa7c3de00   | sysTid=8565 nice=0 cgrp=default sched=1073741825/1 handle=0xa8222dc0   | state=S schedstat=( 421005565 19181973 433 ) utm=25 stm=16 core=2 HZ=100   | stack=0xbe132000-0xbe134000 stackSize=8192KB   | held mutexes=   at b.a.b.c.a.l.v(:1)   - waiting on <0x077aff1b> (a java.lang.Class<b.a.b.c.a.l>)   at b.a.b.c.f.b.e(:6)   at b.a.b.c.a.k.c(:60)   at b.a.b.c.a.k.a(:6)   at b.a.b.c.a.k.<clinit>(:1)   at com.xxx.storage.cache.g1.V(:2)   at com.xxx.storage.cache.q1.a(:20)   at com.xxx.biz.remote.j.i(:113)   at com.xxx.biz.remote.j.f(:6)   at com.transsnet.xxx.lite.service.PlayerService$m.a(:5)   at com.transsnet.xxx.lite.service.PlayerService$m.onChanged(:1)   at com.jeremyliao.liveeventbus.LiveEventBus$ObserverWrapper.onChanged(LiveEventBus.java:3)   at androidx.lifecycle.LiveData.considerNotify(LiveData.java:6)   at androidx.lifecycle.LiveData.dispatchingValue(LiveData.java:8)   at androidx.lifecycle.LiveData.setValue(LiveData.java:4)   at androidx.lifecycle.MutableLiveData.setValue(MutableLiveData.java:1)   at com.jeremyliao.liveeventbus.LiveEventBus$LiveEvent.postInternal(LiveEventBus.java:1)   at com.jeremyliao.liveeventbus.LiveEventBus$LiveEvent.access$1800(LiveEventBus.java:1)   at com.jeremyliao.liveeventbus.LiveEventBus$LiveEvent$PostValueTask.run(LiveEventBus.java:1)   at android.os.Handler.handleCallback(Handler.java:883)   at android.os.Handler.dispatchMessage(Handler.java:100)   at android.os.Looper.loop(Looper.java:214)   at android.app.ActivityThread.main(ActivityThread.java:7669)   at java.lang.reflect.Method.invoke(Native method)   at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:590)   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:936)

这里需要区分一下Blocked和Waiting状态的区别

  • BLOCKED 状态表示线程在等待获取监视器锁,它可能是由于同步块或同步方法的争用导致的。

  • WAITING 状态表示线程在等待其他线程的通知或中断,它可能是由于调用了 wait()、join() 等方法导致的。

这两种状态都属于线程的阻塞状态,但产生的原因和等待的条件是不同的。BLOCKED 状态是由于争夺锁而导致的,而 WAITING 状态是由于线程主动等待的情况。

  • Runnalbe状态

Runnable,显而易见,表示线程正在运行,示例如下:

"main" prio=5 tid=1 Runnable   | group="main" sCount=0 dsCount=0 flags=0 obj=0x722c7448 self=0xb0f3de00   | sysTid=30043 nice=0 cgrp=default sched=1073741825/1 handle=0xb1539dc0   | state=R schedstat=( 6367575122 32733935 3055 ) utm=565 stm=70 core=0 HZ=100   | stack=0xbe581000-0xbe583000 stackSize=8192KB   | held mutexes= "mutator lock"(shared held)   native: #00 pc 00303e8b  /apex/com.android.runtime/lib/libart.so (art::DumpNativeStack(std::\_\_1::basic\_ostream<char, std::\_\_1::char\_traits<char>>&, int, BacktraceMap*, char const*, art::ArtMethod*, void*, bool)+78)   native: #01 pc 003af52b  /apex/com.android.runtime/lib/libart.so (art::Thread::DumpStack(std::\_\_1::basic\_ostream<char, std::\_\_1::char\_traits<char>>&, bool, BacktraceMap*, bool) const+358)   native: #02 pc 003abb73  /apex/com.android.runtime/lib/libart.so (art::Thread::Dump(std::\_\_1::basic\_ostream<char, std::\_\_1::char\_traits<char>>&, bool, BacktraceMap*, bool) const+34)   native: #03 pc 003c4ccb  /apex/com.android.runtime/lib/libart.so (art::DumpCheckpoint::Run(art::Thread*)+606)   native: #04 pc 003aff61  /apex/com.android.runtime/lib/libart.so (art::Thread::RunCheckpointFunction()+128)   native: #05 pc 0041ff45  /apex/com.android.runtime/lib/libart.so (art::JniMethodFastEnd(unsigned int, art::Thread*)+40)   at java.lang.System.arraycopy(Native method)   at d.a.i.put(:16)   at com.bumptech.glide.util.CachedHashCodeArrayMap.put(CachedHashCodeArrayMap.java:2)   at com.bumptech.glide.request.BaseRequestOptions.transform(BaseRequestOptions.java:23)   at com.bumptech.glide.request.BaseRequestOptions.transform(BaseRequestOptions.java:15)   at com.bumptech.glide.request.BaseRequestOptions.optionalTransform(BaseRequestOptions.java:4)   at com.bumptech.glide.request.BaseRequestOptions.scaleOnlyTransform(BaseRequestOptions.java:3)   at com.bumptech.glide.request.BaseRequestOptions.optionalScaleOnlyTransform(BaseRequestOptions.java:1)   at com.bumptech.glide.request.BaseRequestOptions.optionalFitCenter(BaseRequestOptions.java:1)   at com.xxx.util.glide.b.N(:1)   at com.xxx.util.glide.b.optionalFitCenter(:1)   at com.bumptech.glide.RequestBuilder.into(RequestBuilder.java:22)   at e.a.b.b.b.e(:46)   at e.a.b.b.b.g(:2)   at com.xxx.kit.function.HomeLastPlayedView.f(HomeLastPlayedView.java:4)   at com.xxx.kit.function.HomeLastPlayedView.b(HomeLastPlayedView.java:29)   at com.xxx.kit.function.HomeLastPlayedView.g(HomeLastPlayedView.java:12)   at com.xxx.ui.home.c.n1.A1(:4)   at com.xxx.ui.home.c.n1.z1(:5)   at com.xxx.ui.home.c.n1.D(:1)   at com.chad.library.adapter.base.m.n0(:4)   at com.chad.library.adapter.base.m.o0(:2)   at com.chad.library.adapter.base.m.onBindViewHolder(:2)   at androidx.recyclerview.widget.RecyclerView$h.bindViewHolder(:8)  ...省略View绘制流程堆栈...   at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1840)   at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:7937)   at android.view.Choreographer$CallbackRecord.run(Choreographer.java:980)   at android.view.Choreographer.doCallbacks(Choreographer.java:804)   at android.view.Choreographer.doFrame(Choreographer.java:739)   at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:965)   at android.os.Handler.handleCallback(Handler.java:883)   at android.os.Handler.dispatchMessage(Handler.java:100)   at android.os.Looper.loop(Looper.java:264)   at android.app.ActivityThread.main(ActivityThread.java:7605)   at java.lang.reflect.Method.invoke(Native method)   at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:492)   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:980)
  • native状态

native状态其实就是对应java中的Runnable状态,它表明正在执行native代码,也就是通过jni执行底层C语言方法块

"main" prio=5 tid=1 Native   | group="main" sCount=1 dsCount=0 flags=1 obj=0x71356448 self=0xad23de00   | sysTid=5281 nice=0 cgrp=default sched=1073741825/1 handle=0xad781dc0   | state=S schedstat=( 513647352614 2827621143 157445 ) utm=45497 stm=5867 core=0 HZ=100   | stack=0xbe6c9000-0xbe6cb000 stackSize=8192KB   | held mutexes=   kernel: (couldn't read /proc/self/task/5281/stack)   native: #00 pc 0005a16c  /apex/com.android.runtime/lib/bionic/libc.so (syscall+28)   native: #01 pc 0005f58d  /apex/com.android.runtime/lib/bionic/libc.so (\_\_futex\_wait\_ex(void volatile*, bool, int, bool, timespec const*)+88)   native: #02 pc 000a58ad  /apex/com.android.runtime/lib/bionic/libc.so (pthread\_cond\_wait+32)   native: #03 pc 0004dd89  /system/lib/libc++.so (std::\_\_1::condition\_variable::wait(std::\_\_1::unique\_lock<std::\_\_1::mutex>&)+8)   native: #04 pc 0004fbfb  /system/lib/libc++.so (std::\_\_1::\_\_assoc\_sub\_state::copy()+54)   native: #05 pc 0004fdd1  /system/lib/libc++.so (std::__1::future<void>::get()+10)   native: #06 pc 0034e119  /system/lib/libhwui.so (android::uirenderer::renderthread::RenderProxy::destroyHardwareResources()+144)   at android.graphics.HardwareRenderer.nDestroyHardwareResources(Native method)   at android.graphics.HardwareRenderer.clearContent(HardwareRenderer.java:505)   at android.view.ThreadedRenderer.destroyHardwareResources(ThreadedRenderer.java:456)   at android.view.ViewRootImpl.destroyHardwareRenderer(ViewRootImpl.java:7503)   at android.view.ViewRootImpl.dispatchDetachedFromWindow(ViewRootImpl.java:4362)   at android.view.ViewRootImpl.doDie(ViewRootImpl.java:7424)   - locked <0x02d4ecb5> (a android.view.ViewRootImpl)   at android.view.ViewRootImpl.die(ViewRootImpl.java:7397)   at android.view.WindowManagerGlobal.removeViewLocked(WindowManagerGlobal.java:490)   at android.view.WindowManagerGlobal.removeView(WindowManagerGlobal.java:428)   - locked <0x0dc61f4a> (a java.lang.Object)   at android.view.WindowManagerImpl.removeViewImmediate(WindowManagerImpl.java:126)   at android.app.Dialog.dismissDialog(Dialog.java:389)   at android.app.Dialog.dismiss(Dialog.java:371)   at com.xxx.kit.function.CommonDialog$35.onClick(CommonDialog.java:957)   at android.view.View.performClick(View.java:7156)   at android.view.View.performClickInternal(View.java:7129)   at android.view.View.onKeyUp(View.java:14212)   at android.view.KeyEvent.dispatch(KeyEvent.java:2833)   at android.view.View.dispatchKeyEvent(View.java:13398)   at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1921)   at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1931)   at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1931)   at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1931)   at com.android.internal.policy.DecorView.superDispatchKeyEvent(DecorView.java:457)   at com.android.internal.policy.PhoneWindow.superDispatchKeyEvent(PhoneWindow.java:1849)   at android.app.Dialog.dispatchKeyEvent(Dialog.java:825)   at com.android.internal.policy.DecorView.dispatchKeyEvent(DecorView.java:371)   at android.view.ViewRootImpl$ViewPostImeInputStage.processKeyEvent(ViewRootImpl.java:5640)   at android.view.ViewRootImpl$ViewPostImeInputStage.onProcess(ViewRootImpl.java:5503)   at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:5006)   at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:5059)   at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:5025)   at android.view.ViewRootImpl$AsyncInputStage.forward(ViewRootImpl.java:5165)   at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:5033)   at android.view.ViewRootImpl$AsyncInputStage.apply(ViewRootImpl.java:5222)   at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:5006)   at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:5059)   at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:5025)   at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:5033)   at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:5006)   at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:5059)   at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:5025)   at android.view.ViewRootImpl$AsyncInputStage.forward(ViewRootImpl.java:5198)   at android.view.ViewRootImpl$ImeInputStage.onFinishedInputEvent(ViewRootImpl.java:5363)   at android.view.inputmethod.InputMethodManager$PendingEvent.run(InputMethodManager.java:3064)   at android.view.inputmethod.InputMethodManager.invokeFinishedInputEventCallback(InputMethodManager.java:2607)   at android.view.inputmethod.InputMethodManager.finishedInputEvent(InputMethodManager.java:2598)   at android.view.inputmethod.InputMethodManager$ImeInputEventSender.onInputEventFinished(InputMethodManager.java:3041)   at android.view.InputEventSender.dispatchInputEventFinished(InputEventSender.java:143)   at android.os.MessageQueue.nativePollOnce(Native method)   at android.os.MessageQueue.next(MessageQueue.java:336)   at android.os.Looper.loop(Looper.java:215)   at android.app.ActivityThread.main(ActivityThread.java:7605)   at java.lang.reflect.Method.invoke(Native method)   at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:492)   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:980)

了解完了线程的各个状态,我们就对整个anr文件有了一定了解和分析能力了.我们只要再结合堆栈信息,相信我们一定能找我anr所在位置.

接下来我们以上次发生的问题为例,来分析一下,anr发生的原因和位置.

我们可以看到,主线程当前正处于Rnnable状态,说明主线代码整个执行,并没有被其它线程阻塞,那它为什么会发生anr呢?

很容易想到,主线程中肯定是有什么耗时操作影响了其他UI的绘制以及触摸事件的响应.

这里我们可以想到的耗时操作有.

  1. 有耗时的遍历

  2. 有IO操作

  3. 有十分频繁的某个UI的绘制

那我们根据主线程中的堆栈信息可以知道,anr发生的时候,ViewPager正在执行populate计算,这个方法如果有阅读过源码经验的同学,肯定知道它是帮助ViewPager在滑动或者切换时候更新ViewPager左右缓存用的,虽然这里我们没有看到堆栈信息里面有我们自己写的代码,但是我们可以借助debug工具,断点到calculatePageOffsets这个方法,从这个方法里我们根据断点信息,找到了我们代码中可能出现anr位置,如下:

从上面的断点我们可以看到,JDAutoImagerAdapter调用了getCount方法,JDAutoImagerAdapter这个方法是我们自己写的类,而且我们重写了这个类,那我再看一下,我们是如何重写的

我们看到我们重写的时候我们把count设置了int最大值,我们把count设置成了这么大,那有没有可能我们设置成了最大值后,在populate方法中产生一个耗时的循环导致anr的呢,事实的确如此.最终导致anr的原因就是主线程卡在了populate中的一个for循环中,它需要循环的次数是int的最大值除以2,相当于5亿多次.具体ViewPager为什么会这么设置,又是怎么导致这样的anr产生的,这里就不再展开了.毕竟这篇文章主要是为了向大家介绍定位anr的方法.有兴趣的同学可以看一下下面链接的文档进行深究 https://www.jianshu.com/p/d465307accff

从这里我们最终可以确定导致此次anr事件的原因是主线程中有耗时的遍历导致的.至此,整个anr过程分析完成.

4.如何避免anr的产生

了解完了如何定位和分析anr.当然,未雨绸缪才是我们最该做的事情,因此了解如何高效编码才能降低我们产生anr的概率才是我们需要长期努力的方向.下面给予一些建议:

  • 绝对不要在主线程上进行复杂耗时的操作,比如说发送接收网络数据、进行大量计算、操作数据库、读写文件等,统统采用异步操作

  • Service中的耗时操作最好也是采用异步任务

  • 在设计及代码编写阶段避免出现出现死锁、死循环等不恰当情况

  • 使用一些避免、检测ANR的工具

StrictMode:用来检测代码中是否存在违规操作的工具类(检测主线程是否存在耗时操作)

BlockCanary:用来监控应用主线程的卡顿

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

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

相关文章

css3 2D与3D转换

css3 2D与3D转换 前言2D变形旋转变形 rotate()transform-origin属性 缩放变形 scale()斜切变形 skew()位移变形 translate() 3D变形3D旋转 rotateX() | rotateY()perspective属性 空间移动 制作一个正方体结语 前言 网页设计不再局限于平面&#xff0c;而是充满了立体感和动态…

【ESP32接入语言大模型之智谱清言】

1. 智谱清言 讲解视频&#xff1a; 随着人工智能技术的不断发展&#xff0c;自然语言处理领域也得到了广泛的关注和应用。智谱清言作为千亿参数对话模型 基于ChatGLM2模型开发&#xff0c;支持多轮对话&#xff0c;具备内容创作、信息归纳总结等能力。可以快速注册体验中国版…

Linux系统使用超详细(十)~vi/vim命令①

vi/vim命令有很多&#xff0c;其实只有少数的用法对于我们日常工作中起到了很大帮助&#xff0c;但是既然我选择梳理Linux的学习笔记&#xff0c;那么一定全力把自己的理解和学习笔记的内容认真整理汇总&#xff0c;内容或许有错误&#xff0c;还请发现的C友们发现了及时指出。…

小程序基础学习(发送请求)

原理 通过js发起wx.request的方法发送请求并接受相应数据 实例&#xff08;一&#xff09; 参数&#xff1a; url:请求网址地址&#xff0c; success:请求成功执行的函数&#xff0c; fail:请求失败执行的函数 请求返回的数据 实例&#xff08;二&#xff09; 参数&#xff1…

如何用LLM和自有知识库搭建智能agent?

用LangChain建立知识库&#xff0c;文末中也推荐其他方案。 项目源码&#xff1a;ChatPDF实现 LangChain Indexes使用 对加载的内容进行索引&#xff0c;在indexes中提供了一些功能&#xff1a; Document Loaders&#xff0c;加载文档Text Splitters&#xff0c;文档切分V…

Codeforces Round 768 (Div. 1) D. Flipping Range(思维题 等价类性质 dp)

题目 思路来源 官方题解 洛谷题解 题解 可操作的最短区间长度肯定是gcd&#xff0c;记为g&#xff0c;然后考虑如何dp 考虑g个等价类&#xff0c;每个等价类i,ig,i2*g,... 每次翻转长度为g的区间&#xff0c;会同时影响到g个等价类总的翻转的奇偶性&#xff0c; 性质一&…

Puppeteer让你网页操作更简单(2)抓取数据

Puppeteer让你网页操作更简单(1)屏幕截图】 示例2 —— 让我们抓取一些数据 现在您已经了解了Headless Chrome和Puppeteer的工作原理基础知识,让我们看一个更复杂的示例,其中我们实际上可以抓取一些数据。 首先,请查看此处的Puppeteer API文档。如您所见,有大量不同的方法我…

Python教程39:使用turtle画今天日期

---------------turtle源码集合--------------- Python教程36&#xff1a;海龟画图turtle写春联 Python源码35&#xff1a;海龟画图turtle画中国结 Python源码31&#xff1a;海龟画图turtle画七道彩虹 Python源码30&#xff1a;海龟画图turtle画紫色的小熊 Python源码29&a…

10.9.2 std::function 非OO的多态实现 Page185~187

源代码&#xff1a; #include <iostream> #include <functional> #include <list>using namespace std;//使用function模板类定义一个类型&#xff0c; //该类型要求作为T的 //函数类型是参数是string,返回值是void typedef std::function <void (std::s…

MySQL-多表联合查询

&#x1f389;欢迎您来到我的MySQL基础复习专栏 ☆* o(≧▽≦)o *☆哈喽~我是小小恶斯法克&#x1f379; ✨博客主页&#xff1a;小小恶斯法克的博客 &#x1f388;该系列文章专栏&#xff1a;重拾MySQL &#x1f379;文章作者技术和水平很有限&#xff0c;如果文中出现错误&am…

【算法实验】实验1

实验1-1 斐波那契数 【问题描述】斐波那契数 &#xff08;通常用 F(n) 表示&#xff09;形成的序列称为 斐波那契数列 。该数列由 0 和 1 开始&#xff0c;后面的每一项数字都是前面两项数字的和。 定义&#xff1a;F(0) 0, F(1) 1, F(n) F(n-1) F(n-2) 其中n>1 要求计…

1、计算机的硬件组成、校验码

计算机的硬件组成 CPU的组成与功能 CPU的组成 CPU主要是由运算器、控制器、寄存器组和内部总线等部件组成 CPU的功能 1、程序控制通过执行指令来控制程序的执行顺序2、操作功能一条指令的实现需要若干操作信号配合&#xff0c;控制相应的部件完成相对应的操作3、时间控制对…

idea2020.1 x64实现git的push

今天还有点时间&#xff0c;顺便写一下这个。 我这边只说一下远程仓库&#xff08;gitee&#xff09;的push 点击之后会弹出 点击&#xff0c;弹出 输入你定义的远程仓库名&#xff08;自己起&#xff09;&#xff0c;以及url&#xff0c;url由下面获取 在你的gitee创建好仓库…

Marin说PCB之传输线损耗---趋肤效应和导体损耗01

大家在做RF上的PCB走线或者是车载相机的上走线的时候经常会听那些硬件工程师们说你这个走线一定要保证50欧姆的阻抗匹配啊&#xff0c;还有就是记得加粗走做隔层参考。 有的公司的EE硬件同事会很贴心的把RF走线的注意事项给你备注在原理图上或者是layoutguide上&#xff0c;遇到…

Rust-函数

简介 Rust的函数使用关键字fn开头。 函数可以有一系列的输入参数&#xff0c;还有一个返回类型。 函数体包含一系列的语句(或者表达式)。 函数返回可以使用return语句&#xff0c;也可以使用表达式。 Rust编写的可执行程序的入口就是fn main()函数。 以下是一个函数的示例…

Puppeteer让你网页操作更简单(1)屏幕截图

网页自动化设计爬虫工具 中就使用了Puppeteer进行对网页自动化处理&#xff0c;今天就来看看它是什么东西&#xff01; 我们将学习什么? 在本教程中,您将学习如何使用JavaScript自动化和抓取 web。 为此,我们将使用Puppeteer。 Puppeteer是一个Node库API,允许我们控制无头Ch…

在Eureka中注册多个服务(根据本地主机端口号区分)

这篇文章主要介绍如何在Eureka注册中心内注册多个EurekaServer服务端 建项目 创建一个Maven项目&#xff0c;在里面创建三个小的Maven空项目&#xff0c;具体结构如下。 EurekaServer3&#xff0c;即外面的大模块&#xff0c;为了清楚的观察项目结构&#xff0c;我将其src文…

Python data Structures: Strings, Files, Lists

Python data Structures 这门课是在coursera上由美国密歇根大学开展的公开课。python零基础可以选择course1,这是course2&#xff0c;我用来复习和补充的。本章将复习字符串、文件、列表。 Python是这样的&#xff0c;没学一次都有新的发现&#xff0c;补充以前遗漏的地方。 …

助力工业园区作业违规行为检测预警,基于YOLOv7【tiny/l/x】不同系列参数模型开发构建工业园区场景下作业人员违规行为检测识别系统

在很多工业园区生产作业场景下保障合规合法进行作业生产操作&#xff0c;对于保护工人生命安全降低安全隐患有着非常重要的作用&#xff0c;但是往往在实际的作业生产中&#xff0c;因为一个安全观念的淡薄或者是粗心大意&#xff0c;对于纪律约束等意思薄弱&#xff0c;导致在…

开发er们必知的Git命令

Git和GitHub是每位软件工程师都必须了解的最基本的东西。这些工具是开发人员日常工作的组成部分,因为我们每天都要与它们进行交互。熟练掌握Git不仅能简化您的生活,还能显著提高生产力。在这篇博文中,我们将探索一组命令,这些命令将大大提高您的生产力。随着您对这些命令的掌握…