我有一个应用程序,一直在一个特定组件中获得ANR,我无法弄清楚是什么占用了cpu时间。最近我得到了两种ANR,原生和等待。
DALVIK THREADS:
"main" prio=5 tid=1 NATIVE
| group="main" sCount=1 dsCount=0 s=N obj=0x4001d8c0 self=0xccc8
| sysTid=10569 nice=0 sched=0/0 cgrp=default handle=-1345017816
| schedstat=( 9041503981 6690216078 17225 )
at android.view.Surface.lockCanvasNative(Native Method)
at android.view.Surface.lockCanvas(Surface.java:314)
at android.view.SurfaceView$3.internalLockCanvas(SurfaceView.java:773)
at android.view.SurfaceView$3.lockCanvas(SurfaceView.java:756)
at com.dane.hud.Surfaceview$DrawThread.run(Surfaceview.java:776)
at com.dane.hud.Surfaceview.onTouchEvent(Surfaceview.java:322)
at android.view.View.dispatchTouchEvent(View.java:3766)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java:1671)
at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1107)
at android.app.Activity.dispatchTouchEvent(Activity.java:2086)
at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:1655)
at android.view.ViewRoot.handleMessage(ViewRoot.java:1785)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:123)
at android.app.ActivityThread.main(ActivityThread.java:4627)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:521)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:858)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
at dalvik.system.NativeStart.main(Native Method)
"Binder Thread #3" prio=5 tid=9 NATIVE
| group="main" sCount=1 dsCount=0 s=N obj=0x44821de0 self=0x2b9b78
| sysTid=10585 nice=0 sched=0/0 cgrp=default handle=2857768
| schedstat=( 726806597 708740243 2395 )
at dalvik.system.NativeStart.run(Native Method)
"Binder Thread #2" prio=5 tid=6 NATIVE
| group="main" sCount=1 dsCount=0 s=N obj=0x4476c5e8 self=0x13cbf8
| sysTid=10574 nice=0 sched=0/0 cgrp=default handle=1213280
| schedstat=( 767669649 687500005 2512 )
at dalvik.system.NativeStart.run(Native Method)
"Binder Thread #1" prio=5 tid=5 NATIVE
| group="main" sCount=1 dsCount=0 s=N obj=0x4476b310 self=0x126f90
| sysTid=10573 nice=0 sched=0/0 cgrp=default handle=1208144
| schedstat=( 749938913 622802748 2497 )
at dalvik.system.NativeStart.run(Native Method)
"Compiler" daemon prio=5 tid=4 VMWAIT
| group="system" sCount=1 dsCount=0 s=N obj=0x447652a0 self=0x125b80
| sysTid=10572 nice=0 sched=0/0 cgrp=default handle=1251720
| schedstat=( 999145549 669586193 5169 )
at dalvik.system.NativeStart.run(Native Method)
"Signal Catcher" daemon prio=5 tid=3 RUNNABLE
| group="system" sCount=0 dsCount=0 s=N obj=0x447651e8 self=0x121988
| sysTid=10571 nice=0 sched=0/0 cgrp=default handle=1251656
| schedstat=( 16204836 16693113 8 )
at dalvik.system.NativeStart.run(Native Method)
"HeapWorker" daemon prio=5 tid=2 VMWAIT
| group="system" sCount=1 dsCount=0 s=N obj=0x431a9650 self=0x1317c0
| sysTid=10570 nice=0 sched=0/0 cgrp=default handle=1172520
| schedstat=( 28203003083 34572570617 10984 )
at dalvik.system.NativeStart.run(Native Method)
和
DALVIK THREADS:
"main" prio=5 tid=1 WAIT
| group="main" sCount=1 dsCount=0 s=N obj=0x4001d8c0 self=0xccc8
| sysTid=2428 nice=0 sched=0/0 cgrp=default handle=-1345017816
| schedstat=( 5054412632 4232574644 11781 )
at java.lang.Object.wait(Native Method)
- waiting on <0x4001d950> (a java.lang.VMThread)
at java.lang.Thread.parkFor(Thread.java:1535)
at java.lang.LangAccessImpl.parkFor(LangAccessImpl.java:48)
at sun.misc.Unsafe.park(Unsafe.java:317)
at java.util.concurrent.locks.LockSupport.park(LockSupport.java:131)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.parkAndCheckInterrupt(AbstractQueuedSynchronizer.java:790)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireQueued(AbstractQueuedSynchronizer.java:823)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquire(AbstractQueuedSynchronizer.java:1153)
at java.util.concurrent.locks.ReentrantLock$NonfairSync.lock(ReentrantLock.java:185)
at java.util.concurrent.locks.ReentrantLock.lock(ReentrantLock.java:261)
at android.view.SurfaceView$3.internalLockCanvas(SurfaceView.java:764)
at android.view.SurfaceView$3.lockCanvas(SurfaceView.java:756)
at com.dane.hud.Surfaceview$DrawThread.run(Surfaceview.java:776)
at com.dane.hud.Surfaceview.onTouchEvent(Surfaceview.java:319)
at android.view.View.dispatchTouchEvent(View.java:3766)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:936)
at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java:1671)
at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1107)
at android.app.Activity.dispatchTouchEvent(Activity.java:2086)
at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:1655)
at android.view.ViewRoot.handleMessage(ViewRoot.java:1785)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:123)
at android.app.ActivityThread.main(ActivityThread.java:4627)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:521)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:858)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
at dalvik.system.NativeStart.main(Native Method)
"Thread-448" prio=5 tid=16 WAIT
| group="main" sCount=1 dsCount=0 s=N obj=0x447e4b40 self=0x2e5960
| sysTid=2892 nice=0 sched=0/0 cgrp=default handle=3037856
| schedstat=( 495635974 519531260 463 )
at java.lang.Object.wait(Native Method)
- waiting on <0x447eeb10> (a java.lang.VMThread)
at java.lang.Thread.parkFor(Thread.java:1535)
at java.lang.LangAccessImpl.parkFor(LangAccessImpl.java:48)
at sun.misc.Unsafe.park(Unsafe.java:317)
at java.util.concurrent.locks.LockSupport.park(LockSupport.java:131)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.parkAndCheckInterrupt(AbstractQueuedSynchronizer.java:790)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireQueued(AbstractQueuedSynchronizer.java:823)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquire(AbstractQueuedSynchronizer.java:1153)
at java.util.concurrent.locks.ReentrantLock$NonfairSync.lock(ReentrantLock.java:185)
at java.util.concurrent.locks.ReentrantLock.lock(ReentrantLock.java:261)
at android.view.SurfaceView$3.internalLockCanvas(SurfaceView.java:764)
at android.view.SurfaceView$3.lockCanvas(SurfaceView.java:756)
at com.dane.hud.Surfaceview$DrawThread.run(Surfaceview.java:776)
at com.dane.hud.HUD.updateWithNewLocation(HUD.java:639)
at com.dane.hud.HUD.access$3(HUD.java:618)
at com.dane.hud.HUD$updateEverythingFromGPS.run(HUD.java:573)
"Thread-447" prio=5 tid=15 WAIT
| group="main" sCount=1 dsCount=0 s=N obj=0x447fb5f0 self=0x2bd690
| sysTid=2890 nice=0 sched=0/0 cgrp=default handle=2873296
| schedstat=( 636688219 727813733 545 )
at java.lang.Object.wait(Native Method)
- waiting on <0x447fb780> (a java.lang.VMThread)
at java.lang.Thread.parkFor(Thread.java:1535)
at java.lang.LangAccessImpl.parkFor(LangAccessImpl.java:48)
at sun.misc.Unsafe.park(Unsafe.java:317)
at java.util.concurrent.locks.LockSupport.park(LockSupport.java:131)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.parkAndCheckInterrupt(AbstractQueuedSynchronizer.java:790)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireQueued(AbstractQueuedSynchronizer.java:823)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquire(AbstractQueuedSynchronizer.java:1153)
at java.util.concurrent.locks.ReentrantLock$NonfairSync.lock(ReentrantLock.java:185)
at java.util.concurrent.locks.ReentrantLock.lock(ReentrantLock.java:261)
at android.view.SurfaceView$3.internalLockCanvas(SurfaceView.java:764)
at android.view.SurfaceView$3.lockCanvas(SurfaceView.java:756)
at com.dane.hud.Surfaceview$DrawThread.run(Surfaceview.java:776)
at com.dane.hud.HUD.updateWithNewLocation(HUD.java:639)
at com.dane.hud.HUD.access$3(HUD.java:618)
at com.dane.hud.HUD$updateEverythingFromGPS.run(HUD.java:573)
"Thread-446" prio=5 tid=14 WAIT
| group="main" sCount=1 dsCount=0 s=N obj=0x44823a68 self=0x2ccac0
| sysTid=2889 nice=0 sched=0/0 cgrp=default handle=2874496
| schedstat=( 774780280 777496325 693 )
at java.lang.Object.wait(Native Method)
- waiting on <0x44786fe8> (a java.lang.VMThread)
at java.lang.Thread.parkFor(Thread.java:1535)
at java.lang.LangAccessImpl.parkFor(LangAccessImpl.java:48)
at sun.misc.Unsafe.park(Unsafe.java:317)
at java.util.concurrent.locks.LockSupport.park(LockSupport.java:131)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.parkAndCheckInterrupt(AbstractQueuedSynchronizer.java:790)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireQueued(AbstractQueuedSynchronizer.java:823)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquire(AbstractQueuedSynchronizer.java:1153)
at java.util.concurrent.locks.ReentrantLock$NonfairSync.lock(ReentrantLock.java:185)
at java.util.concurrent.locks.ReentrantLock.lock(ReentrantLock.java:261)
at android.view.SurfaceView$3.internalLockCanvas(SurfaceView.java:764)
at android.view.SurfaceView$3.lockCanvas(SurfaceView.java:756)
at com.dane.hud.Surfaceview$DrawThread.run(Surfaceview.java:776)
at com.dane.hud.HUD.updateWithNewLocation(HUD.java:639)
at com.dane.hud.HUD.access$3(HUD.java:618)
at com.dane.hud.HUD$updateEverythingFromGPS.run(HUD.java:573)
"Thread-445" prio=5 tid=13 WAIT
| group="main" sCount=1 dsCount=0 s=N obj=0x447f2288 self=0x2e2a80
| sysTid=2888 nice=0 sched=0/0 cgrp=default handle=3026032
| schedstat=( 831634516 815490735 696 )
at java.lang.Object.wait(Native Method)
- waiting on <0x447f2418> (a java.lang.VMThread)
at java.lang.Thread.parkFor(Thread.java:1535)
at java.lang.LangAccessImpl.parkFor(LangAccessImpl.java:48)
at sun.misc.Unsafe.park(Unsafe.java:317)
at java.util.concurrent.locks.LockSupport.park(LockSupport.java:131)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.parkAndCheckInterrupt(AbstractQueuedSynchronizer.java:790)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireQueued(AbstractQueuedSynchronizer.java:823)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquire(AbstractQueuedSynchronizer.java:1153)
at java.util.concurrent.locks.ReentrantLock$NonfairSync.lock(ReentrantLock.java:185)
at java.util.concurrent.locks.ReentrantLock.lock(ReentrantLock.java:261)
at android.view.SurfaceView$3.internalLockCanvas(SurfaceView.java:764)
at android.view.SurfaceView$3.lockCanvas(SurfaceView.java:756)
at com.dane.hud.Surfaceview$DrawThread.run(Surfaceview.java:776)
at com.dane.hud.HUD.updateWithNewLocation(HUD.java:639)
at com.dane.hud.HUD.access$3(HUD.java:618)
at com.dane.hud.HUD$updateEverythingFromGPS.run(HUD.java:573)
"Thread-444" prio=5 tid=12 WAIT
| group="main" sCount=1 dsCount=0 s=N obj=0x447c97c0 self=0x2c4910
| sysTid=2887 nice=0 sched=0/0 cgrp=default handle=2780792
| schedstat=( 1249114988 1195007328 1038 )
at java.lang.Object.wait(Native Method)
- waiting on <0x447cd918> (a java.lang.VMThread)
at java.lang.Thread.parkFor(Thread.java:1535)
at java.lang.LangAccessImpl.parkFor(LangAccessImpl.java:48)
at sun.misc.Unsafe.park(Unsafe.java:317)
at java.util.concurrent.locks.LockSupport.park(LockSupport.java:131)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.parkAndCheckInterrupt(AbstractQueuedSynchronizer.java:790)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireQueued(AbstractQueuedSynchronizer.java:823)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquire(AbstractQueuedSynchronizer.java:1153)
at java.util.concurrent.locks.ReentrantLock$NonfairSync.lock(ReentrantLock.java:185)
at java.util.concurrent.locks.ReentrantLock.lock(ReentrantLock.java:261)
at android.view.SurfaceView$3.internalLockCanvas(SurfaceView.java:764)
at android.view.SurfaceView$3.lockCanvas(SurfaceView.java:756)
at com.dane.hud.Surfaceview$DrawThread.run(Surfaceview.java:776)
at com.dane.hud.HUD.updateWithNewLocation(HUD.java:639)
at com.dane.hud.HUD.access$3(HUD.java:618)
at com.dane.hud.HUD$updateEverythingFromGPS.run(HUD.java:573)
"Thread-443" prio=5 tid=11 VMWAIT JIT
| group="main" sCount=1 dsCount=0 s=Y obj=0x4480cd28 self=0x2d27b0
| sysTid=2886 nice=0 sched=0/0 cgrp=default handle=2959600
| schedstat=( 1636657703 1482391360 1374 )
at java.util.concurrent.locks.AbstractQueuedSynchronizer.addWaiter(AbstractQueuedSynchronizer.java:~562)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquire(AbstractQueuedSynchronizer.java:1153)
at java.util.concurrent.locks.ReentrantLock$NonfairSync.lock(ReentrantLock.java:185)
at java.util.concurrent.locks.ReentrantLock.lock(ReentrantLock.java:261)
at android.view.Surfa...
有什么想法吗?我已经将操作移动到新线程,也许我没有调用join();在他们身上。
感谢您的帮助。
答案 0 :(得分:5)
在这里:
http://developer.android.com/guide/practices/design/responsiveness.html
ANR是主线程阻塞的东西,所以总是建议在子线程中执行长操作:
引用:
因此,任何运行的方法 主线程应该做的很少 尽可能。特别是活动 应该做尽可能少的设置 关键的生命周期方法,如 onCreate()和onResume()。潜在 长时间运行的操作,如 网络或数据库操作,或 计算上昂贵的计算 比如调整位图应该是 在子线程中完成(或在案例中 数据库操作,通过 异步请求)。但是,这个 并不代表你的主线程 在等待时应该阻止 子线程完成 - 也不应该 你调用Thread.wait()或 Thread.sleep()方法。而不是阻止 在等待子线程时 完成,你的主线程应该 为子线程提供一个处理程序 回到完成后。 以这种方式设计您的应用程序 将允许你的主线程保持 响应输入,从而避免ANR 由5秒输入引起的对话框 事件超时。这些相同的做法 应该遵循任何其他 显示UI的线程,就像它们一样 也受到相同的超时限制。
我理解这并不容易,但上面的建议是UI不应该在主线程中完成。
另一方面,主线程处于“WAIT”状态,表示使用非异步操作。也许你想尝试async选项 - 它应该导致TIMED_WAIT状态,如下所示:
Android - how do I investigate an ANR?
(主线程正常)。
但就像这里讨论的那样:
你的问题我怀疑是因为位图绘制在lockCanvas()和unlockCanvas()之间花了太长时间 - 也许你可能想要将位图更新分成更小的部分?在同一讨论中还提到了使用JNI / NDK进行位图计算 - 如果在计算过程中花费太长时间,则不应在基于Java的位图更新方法内部进行。
推荐阅读:
http://obviam.net/index.php/the-android-game-loop/
感谢。
答案 1 :(得分:0)
基于日志,我能想到的唯一可能触发ANR的原因是:
据我所知,通常在Java中,每个线程都与2个组件相关联。一个是程序计数器,另一个是垃圾收集器。因此,每个线程都会在整个过程中发生gc监视器。
java.lang.Object.wait(Native Method)- 等待&lt; 0x4001d950&gt; (java.lang.VMThread)
VMThreads理想地用于处理监视gc的这个方面,几乎是并行的。所以我能想到的是,对于你的场景,SurfaceViewis试图获取一个锁但是因为VMThread正在持有锁,尚未释放(可能是由于一些主要的GC操作),因此等待vmthread释放锁
您可能想要检查记录了gc活动的主日志转储,这可能会让您更好地了解为什么gc占用此线程的CPU时间太多,然后应用程序分析可能会有所帮助。检查。