Android中对Log日志文件的分析

一,Bug出现了, 需要“干掉”它 

bug一听挺吓人的,但是只要你懂了,android里的bug是很好解决的,因为android里提供了LOG机制,具体的底层代码,以后在来分析,只要你会看bug,

android里应用开发也就很简单了。

那我们先来看看android里的ANR,怎么出现ANR呢,很简单。

          # adb shell

          # cd data/app

          # monkey   -p  com.xxx.xxx   -v   3000      (com.xxx.xxx是你应用程序的包名,如果想知道monkey详细用法,执行  monkey  help  )

         实际上很多优秀android应用都会出现ANR,比如UC浏览器,360等等,如果你有兴趣可以回去试试,

         这样,ANR出现了。 开始做修改准备工作  ,得到log文件。

有人问log文件在哪儿?
一般在/data/log下面(但是真机才有的)。你可以通过执行命令adb shell进去看看,用pull把log文件拉到你的电脑里
 
好,得到log文件了,我们就准备开始工作了 。 我将详细的log文件上传到附件供大家参考。

下载地址为  http://download.csdn.net/detail/andy_android/3785393


二,Log的种类

          android.util.Log常用的方法有以下六个:Log.v() Log.d() Log.i() Log.w() , Log.e()以及Log.a() (android 4.0新增加的)。根据首字母对应             VERBOSEDEBUG,INFO,WARNERROR,Assert。

          1、Log.v 的调试颜色为黑色的,任何消息都会输出,这里的v代表verbose啰嗦的意思,平时使用就是Log.v("","");

          2、Log.d的输出颜色是蓝色的,仅输出debug调试的意思,但他会输出上层的信息,过滤起来可以通过DDMS的Logcat标签来选择.

          3、Log.i的输出为绿色,一般提示性的消息information,它不会输出Log.v和Log.d的信息,但会显示i、w和e的信息

         4、Log.w的意思为橙色,可以看作为warning警告,一般需要我们注意优化Android代码,同时选择它后还会输出Log.e的信息。

         5、Log.e为红色,可以想到error错误,这里仅显示红色的错误信息,这些错误就需要我们认真的分析,查看栈的信息了。

         6    Log.a为4.0新增加的。

          启动Eclipse,打开DDMS大家可以在LogCat里看见(前提是android 4.0的ADT)

三,如何分析和研究Log文件,如何看日志信息 

Logandroid中的地位非常重要,要是作为一个android程序员不能过分析log这关,算是android没有入门吧。 下面我们就来说说如何处理log文件。

Log分为Fatal和ANR

什么时候会有Log文件的产生?
Log
的产生大家都知道, 大家也都知道通过DDMS来看log(这个就不用说了),但什么时候会产生log文件呢?一般在如下几种情况会产生log文件。 
1
,程序异常退出,uncausedexception      (Fatal)
2
,程序强制关闭,ForceClosed (简称FC)       (Fatal)
3
,程序无响应,ApplicationNo Response(简称ANR)

    ANR出现的情况有以下两种

      A  界面操作按钮的点击等待响应时间超过5秒

      B  HandleMessage回调函数执行超过10秒,BroadcasterReciver里的onRecive()方法处理超过10秒

4,手动生成。


拿到一个日志文件,要分成多段来看。log文件很长,其中包含十几个小单元信息,但不要被吓到,事实上他主要由三大块儿组成 。

1
,系统基本信息,包括 内存,CPU,进程队列,虚拟内存 , 垃圾回收等信息 。------MEMORY INFO (/proc/meminfo) ------
------CPU INFO (top -n 1 -d 1 -m 30 -t) ------
------PROCRANK (procrank) ------
------VIRTUAL MEMORY STATS (/proc/vmstat) ------
------VMALLOC INFO (/proc/vmallocinfo) ------

格式如下:
------MEMORY INFO (/proc/meminfo) ------
MemTotal:        347076 kB
MemFree:          56408 kB
Buffers:           7192 kB
Cached:          104064 kB
SwapCached:           0 kB
Active:          192592 kB
Inactive:         40548 kB
Active(anon):    129040 kB
Inactive(anon):    1104 kB
Active(file):     63552 kB
Inactive(file):   39444 kB
Unevictable:       7112 kB
Mlocked:              0kB
SwapTotal:            0 kB
SwapFree:             0 kB
Dirty:               44kB
Writeback:            0 kB
AnonPages:       129028 kB
Mapped:           73728 kB
Shmem:             1148kB
Slab:             13072kB
SReclaimable:      4564 kB
SUnreclaim:        8508 kB
KernelStack:       3472 kB
PageTables:       12172 kB
NFS_Unstable:         0 kB
Bounce:               0kB
WritebackTmp:         0 kB
CommitLimit:     173536 kB
Committed_AS:   7394524 kB
VmallocTotal:    319488 kB
VmallocUsed:      90752 kB
VmallocChunk:    181252 kB


2
,事件信息, 也是我们主要分析的信息 。
------VMALLOC INFO (/proc/vmallocinfo) ------
------EVENT INFO (/proc/vmallocinfo) ------

格式如下:
------SYSTEM LOG (logcat -b system -v time -d *:v) ------
01-1516:41:43.671 W/PackageManager( 2466): Unknown permissioncom.wsomacp.permission.PROVIDER in package com.android.mms
01-1516:41:43.671 I/ActivityManager( 2466): Force stopping packagecom.android.mms uid=10092
01-1516:41:43.675 I/UsageStats( 2466): Something wrong here, didn't expectcom.sec.android.app.twlauncher to be paused
01-1516:41:44.108 I/ActivityManager( 2466): Start proccom.sec.android.widgetapp.infoalarm for servicecom.sec.android.widgetapp.infoalarm/.engine.DataService: pid=20634uid=10005 gids={3003, 1015, 3002}
01-1516:41:44.175 W/ActivityManager( 2466): Activity pause timeout forHistoryRecord{48589868com.sec.android.app.twlauncher/.Launcher}
01-1516:41:50.864 I/KeyInputQueue( 2466): Input event
01-1516:41:50.866 D/KeyInputQueue( 2466): screenCaptureKeyFlag setting0
01-1516:41:50.882 I/PowerManagerService( 2466): Ulight 0->7|0
01-1516:41:50.882 I/PowerManagerService( 2466): Setting target 2: cur=0.0target=70 delta=4.6666665 nominalCurrentValue=0
01-1516:41:50.882 I/PowerManagerService( 2466): Scheduling lightanimator!
01-1516:41:51.706 D/PowerManagerService( 2466): enableLightSensortrue
01-1516:41:51.929 I/KeyInputQueue( 2466): Input event
01-1516:41:51.933 W/WindowManager( 2466): No focus window, dropping:KeyEvent{action=0 code=26 repeat=0 meta=0 scancode=26mFlags=9}



3
,虚拟机信息, 包括进程的,线程的跟踪信息,这是用来跟踪进程和线程具体点的好地方。 
------VM TRACES JUST NOW (/data/anr/traces.txt.bugreport: 2011-01-1516:49:02) ------
------VM TRACES AT LAST ANR (/data/anr/traces.txt: 2011-01-15 16:49:02)------


格式如下:
-----pid 21161 at 2011-01-15 16:49:01 -----
Cmdline: com.android.mms

DALVIKTHREADS:
"main"prio=5 tid=1 NATIVE
  |group="main" sCount=1 dsCount=0 s=N obj=0x4001d8d0self=0xccc8
  |sysTid=21161 nice=0 sched=0/0 cgrp=default handle=-1345017808
  |schedstat=( 4151552996 5342265329 10995 )
  atandroid.media.MediaPlayer._reset(Native Method)
  atandroid.media.MediaPlayer.reset(MediaPlayer.java:1218)
  atandroid.widget.VideoView.release(VideoView.java:499)
  atandroid.widget.VideoView.access$2100(VideoView.java:50)
  atandroid.widget.VideoView$6.surfaceDestroyed(VideoView.java:489)
  atandroid.view.SurfaceView.reportSurfaceDestroyed(SurfaceView.java:572)
  atandroid.view.SurfaceView.updateWindow(SurfaceView.java:476)
  atandroid.view.SurfaceView.onWindowVisibilityChanged(SurfaceView.java:206)
  atandroid.view.View.dispatchDetachedFromWindow(View.java:6082)
  atandroid.view.ViewGroup.dispatchDetachedFromWindow(ViewGroup.java:1156)
  atandroid.view.ViewGroup.removeAllViewsInLayout(ViewGroup.java:2296)
  atandroid.view.ViewGroup.removeAllViews(ViewGroup.java:2254)
  atcom.android.mms.ui.SlideView.reset(SlideView.java:687)
  atcom.android.mms.ui.SlideshowPresenter.presentSlide(SlideshowPresenter.java:189)
  atcom.android.mms.ui.SlideshowPresenter$3.run(SlideshowPresenter.java:531)
  atandroid.os.Handler.handleCallback(Handler.java:587)
  atandroid.os.Handler.dispatchMessage(Handler.java:92)
  atandroid.os.Looper.loop(Looper.java:123)
  atandroid.app.ActivityThread.main(ActivityThread.java:4627)
  atjava.lang.reflect.Method.invokeNative(Native Method)
  atjava.lang.reflect.Method.invoke(Method.java:521)
  atcom.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:858)
  atcom.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
  atdalvik.system.NativeStart.main(NativeMethod)

---------------------------------------------------------------------------------------------------------------------------------------
闲话少说,我总结了观察log文件的基本步骤。

1,如果是ANR问题, 则搜索“ANR”关键词。 快速定位到关键事件信息 。
2
,如果是ForceClosed和其它异常退出信息,则搜索"Fatal"关键词,快速定位到关键事件信息 。
3
,定位到关键事件信息后, 如果信息不够明确的,再去搜索应用程序包的虚拟机信息,查看具体的进程和线程跟踪的日志,来定位到代码。 

用这种方法,出现问题,根本不需要断点调试, 直接定位到问题,屡试不爽 。 
下面,我们就开始来分析这个例子的log

打开log文件, 由于是ANR错误,因此搜索"ANR",为何要加空格呢,你加上和去掉比较一下就知道了 。可以屏蔽掉不少保存到anr.log文件的无效信息。 

定位到关键的事件信息如下:
01-1516:49:02.433 E/ActivityManager( 2466): ANR in com.android.mms(com.android.mms/.ui.SlideshowActivity)
01-1516:49:02.433 E/ActivityManager( 2466): Reason:keyDispatchingTimedOut
01-1516:49:02.433 E/ActivityManager( 2466): Load: 0.6 / 0.61 / 0.42
01-1516:49:02.433 E/ActivityManager( 2466): CPU usage from 1337225ms to57ms ago:
01-1516:49:02.433 E/ActivityManager( 2466):   sensorserver_ya:8% = 0% user + 8% kernel / faults: 40 minor
......


01-1516:49:02.433 E/ActivityManager( 2466):  -com.android.mms:0% = 0% user + 0% kernel
01-1516:49:02.433 E/ActivityManager( 2466):  -flush-179:8: 0% =0% user + 0% kernel
01-1516:49:02.433 E/ActivityManager( 2466): TOTAL: 25% = 10% user + 14%kernel + 0% iowait + 0% irq + 0% softirq
01-1516:49:02.436 I/        ( 2466):dumpmesg >"/data/log/dumpstate_app_anr.log"


我们用自然语言来描述一下日志,这也算是一种能力吧。 
01-1516:49:02.433 E/ActivityManager( 2466): ANR in com.android.mms(com.android.mms/.ui.SlideshowActivity)
翻译:在16:492433毫秒的时候ActivityManager(进程号为2466)发生了如下错误:com.android.mms包下面的.ui.SlideshowActivity无响应。

01-1516:49:02.433 E/ActivityManager( 2466): Reason:keyDispatchingTimedOut
翻译:原因,keyDispatchingTimeOut-按键分配超时 

01-1516:49:02.433 E/ActivityManager( 2466): Load: 0.6 / 0.61 /0.42
翻译:5分钟,10分钟,15分钟内的平均负载分别为:0.6, 0.61 ,0.42

在这里我们大概知道问题是什么了,结合我们之前的操作流程,我们知道问题是在点击按钮某时候可能处理不过来按钮事件,导致超时无响应。那么现在似乎已经可以进行工作了 。我们知道Activity中是通过重载dispatchTouchEvent(MotionEventev)来处理点击屏幕事件  。然后我们可以顺藤摸瓜,一点点分析去查找原因 。但这样够了么 ?
其实不够, 至少我们不能准确的知道到底问题在哪儿 , 只是猜测,比如这个应用程序中,我就在顺藤摸瓜的时候发现了多个IO操作的地方都在主线程中,可能引起问题,但不好判断到底是哪个  ,所以我们目前掌握的信息还不够。 

于是我们再分析虚拟机信息, 搜索“DalvikThread”关键词,快速定位到本应用程序的虚拟机信息日志,如下:
-----pid 2922 at 2011-01-13 13:51:07 -----
Cmdline: com.android.mms

DALVIKTHREADS:
"main"prio=5 tid=1 NATIVE
  |group="main" sCount=1 dsCount=0 s=N obj=0x4001d8d0self=0xccc8
  |sysTid=2922 nice=0 sched=0/0 cgrp=default handle=-1345017808
  |schedstat=( 3497492306 15312897923 10358 )
  atandroid.media.MediaPlayer._release(Native Method)
  atandroid.media.MediaPlayer.release(MediaPlayer.java:1206)
  atandroid.widget.VideoView.stopPlayback(VideoView.java:196)
  atcom.android.mms.ui.SlideView.stopVideo(SlideView.java:640)
  atcom.android.mms.ui.SlideshowPresenter.presentVideo(SlideshowPresenter.java:443)
  atcom.android.mms.ui.SlideshowPresenter.presentRegionMedia(SlideshowPresenter.java:219)
  atcom.android.mms.ui.SlideshowPresenter$4.run(SlideshowPresenter.java:516)
  atandroid.os.Handler.handleCallback(Handler.java:587)
  atandroid.os.Handler.dispatchMessage(Handler.java:92)
  atandroid.os.Looper.loop(Looper.java:123)
  atandroid.app.ActivityThread.main(ActivityThread.java:4627)
  atjava.lang.reflect.Method.invokeNative(Native Method)
  atjava.lang.reflect.Method.invoke(Method.java:521)
  atcom.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:858)
  atcom.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
  atdalvik.system.NativeStart.main(Native Method)

"BinderThread #3" prio=5 tid=11 NATIVE
  |group="main" sCount=1 dsCount=0 s=N obj=0x4837f808self=0x242280
  |sysTid=3239 nice=0 sched=0/0 cgrp=default handle=2341032
  |schedstat=( 32410506 932842514 164 )
  atdalvik.system.NativeStart.run(Native Method)

"AsyncQueryWorker"prio=5 tid=9 WAIT
  |group="main" sCount=1 dsCount=0 s=N obj=0x482f4b80self=0x253e10
  |sysTid=3236 nice=0 sched=0/0 cgrp=default handle=2432120
  |schedstat=( 3225061 26561350 27 )
  atjava.lang.Object.wait(Native Method)
  -waiting on <0x482f4da8> (a android.os.MessageQueue)
  atjava.lang.Object.wait(Object.java:288)
  atandroid.os.MessageQueue.next(MessageQueue.java:146)
  atandroid.os.Looper.loop(Looper.java:110)
  atandroid.os.HandlerThread.run(HandlerThread.java:60)

"Thread-9"prio=5 tid=8 WAIT
  |group="main" sCount=1 dsCount=0 s=N obj=0x4836e2b0self=0x25af70
  |sysTid=2929 nice=0 sched=0/0 cgrp=default handle=2370896
  |schedstat=( 130248 4389035 2 )
  atjava.lang.Object.wait(Native Method)
  -waiting on <0x4836e240> (a java.util.ArrayList)
  atjava.lang.Object.wait(Object.java:288)
  atcom.android.mms.data.Contact$ContactsCache$TaskStack$1.run(Contact.java:488)
  atjava.lang.Thread.run(Thread.java:1096)

"BinderThread #2" prio=5 tid=7 NATIVE
  |group="main" sCount=1 dsCount=0 s=N obj=0x482f8ca0self=0x130fd0
  |sysTid=2928 nice=0 sched=0/0 cgrp=default handle=1215968
  |schedstat=( 40610049 1837703846 195 )
  atdalvik.system.NativeStart.run(Native Method)

"BinderThread #1" prio=5 tid=6 NATIVE
  |group="main" sCount=1 dsCount=0 s=N obj=0x482f4a78self=0x128a50
  |sysTid=2927 nice=0 sched=0/0 cgrp=default handle=1201352
  |schedstat=( 40928066 928867585 190 )
  atdalvik.system.NativeStart.run(Native Method)

"Compiler"daemon prio=5 tid=5 VMWAIT
  |group="system" sCount=1 dsCount=0 s=N obj=0x482f1348self=0x118960
  |sysTid=2926 nice=0 sched=0/0 cgrp=default handle=1149216
  |schedstat=( 753021350 3774113668 6686 )
  atdalvik.system.NativeStart.run(Native Method)

"JDWP"daemon prio=5 tid=4 VMWAIT
  |group="system" sCount=1 dsCount=0 s=N obj=0x482f12a0self=0x132940
  |sysTid=2925 nice=0 sched=0/0 cgrp=default handle=1255680
  |schedstat=( 2827103 29553323 19 )
  atdalvik.system.NativeStart.run(Native Method)

"SignalCatcher" daemon prio=5 tid=3 RUNNABLE
  |group="system" sCount=0 dsCount=0 s=N obj=0x482f11e8self=0x135988
  |sysTid=2924 nice=0 sched=0/0 cgrp=default handle=1173688
  |schedstat=( 11793815 12456169 7 )
  atdalvik.system.NativeStart.run(Native Method)

"HeapWorker"daemon prio=5 tid=2 VMWAIT
  |group="system" sCount=1 dsCount=0 s=N obj=0x45496028self=0x135848
  |sysTid=2923 nice=0 sched=0/0 cgrp=default handle=1222608
  |schedstat=( 79049792 1520840200 95 )
  atdalvik.system.NativeStart.run(Native Method)

-----end 2922 -----

每一段都是一个线程,当然我们还是看线程号为1的主线程了。通过分析发现关键问题是这样:
  atcom.android.mms.ui.SlideshowPresenter$3.run(SlideshowPresenter.java:531)
定位到代码:
mHandler.post(newRunnable() {
                  public void run() {
                      try {
                          presentRegionMedia(view,(RegionMediaModel) model, dataChanged);
                      } catch (OMADRMException e) {
                          Log.e(TAG, e.getMessage(), e);
                          Toast.makeText(mContext,
                                 mContext.getString(R.string.insufficient_drm_rights),
                                 Toast.LENGTH_SHORT).show();
                      } catch (IOException e){
                          Log.e(TAG, e.getMessage(), e);
                          Toast.makeText(mContext,
                                 mContext.getString(R.string.insufficient_drm_rights),
                                 Toast.LENGTH_SHORT).show();
                      

                      }
                  }

很清楚了,Handler.post方法之后执行时间太长的问题。 继续看presentRegionMedia(view,(RegionMediaModel) model, dataChanged);方法, 发现最终是调用的frameworkMediaPlayer.stop方法。
至此,我们的日志分析算是告一段落。 可以开始思考解决办法了。 

四,如何通过Handler或者多线程来解决某操作执行时间过程的问题。

结合上面的分析,我们知道问题似乎是线程队列中某个操作presentRegionMedia(view,(RegionMediaModel) model, dataChanged);执行时间太长所导致的界面无响应。 因此比较典型的做法当然是控制线程队列 。在这里我们不得不提一下Handler  . 

Handler
Android中是什么样的作用和地位呢?

  1. 线程之间消息传递 ,通过sendMessage方法 。我们通常用来后台子线程向主线程传递消息,主线程接到通知之后做更新界面等操作。

  2. 通过管理消息队列(MessageQueue)来安排计划任务。 这个常常会被人忽略,很多书上也没有提到这个作用。

Handler这个单词中文意思是管理者,处理者的意思。 通过这个意思顾名思义,我们知道这个对象就是个操作对象。那么要操作谁呢?

当然是消息队列(MessageQueue)。Android消息队列类似于Win32队列设计。 都是采用线性结构,先进先出 。其实在智能手机平台很久以前就用这种消息结构了 。比如Palm ,只不过Palm是整个进程共享一个消息队列,而Android是线程为单位的队列罢了。

那么是否每个线程或者子线程都有消息队列呢?

很遗憾,不是的,也没有必要。 在Android中,只有使用了Looper的线程才有消息队列。 当然如果你要简单建立一个有消息队列的线程也很方便,直接使用HandlerThread即可,这个类继承于Thread类。怎么用我就不多说了吧 。你懂的 !

Handler有两种方式来操作消息队列。

一种是通过sendMessage(Message)方法,发送消息体

另一种是通过post(Runnable)方法 , 发送Runnable对象 。

注意:这点请注意,虽然发送方法含参不同 , 但他们使用的是同一个消息队列。 我记得Mars的视频教程上说有两个队列,一个是消息队列,一个是线程队列。 这种说法是错误的 。事实上只有一个消息队列,没有所谓的线程队列。 当然了 ,post(Runnable)也没有启动新的线程,仍然是在当前线程。

注意:还有一种说法,说Handler对象在主线程,这种说法也是错误的, 准确的说是在产生他的线程中 。 虽然常常我们是在主线程产生他的。

那么我们要在Android建立多线程程序该如何做呢?很简单,就是Java的多线程方式。要么实现Runnable接口,要么继承Thread类。

关于线程同步,线程锁定,线程异步,线程池 这些概念也是一样的 。 我就不累述了。


好了,经过一点儿简单的介绍,我们有了一些Handler的基础,现在开始回到我们的问题开始来分析:

mHandler.post(newRunnable() {
                  public void run() {
                      try {
                          presentRegionMedia(view,(RegionMediaModel) model, dataChanged);
                      } catch (OMADRMException e) {
                          Log.e(TAG, e.getMessage(), e);
                          Toast.makeText(mContext,
                                 mContext.getString(R.string.insufficient_drm_rights),
                                 Toast.LENGTH_SHORT).show();
                      } catch (IOException e){
                          Log.e(TAG, e.getMessage(), e);
                          Toast.makeText(mContext,
                                 mContext.getString(R.string.insufficient_drm_rights),
                                 Toast.LENGTH_SHORT).show();
                      

                      }
                  }

从上面这段代码中,我们可以看出,在做播放器控制按钮(比如播放,暂停,停止)等操作的时候, 是通过Handler.post(Runnable)来放到消息队列中, 排序来处理 。 那么之所以这里出现了无响应,很有可能是因为某一项控制操作太耗时或者耗资源。 这时候又接收到新的要处理的消息,就会处理不过来了。 因此我试图让队列中同时只有一个控制播放器按钮的任务在。 我对代码做了如下改动:

Runnabler = new Runnable(){
                  public void run() {
                      try {
                          presentRegionMedia(view,(RegionMediaModel) model, dataChanged);
                      } catch (OMADRMException e) {
                          Log.e(TAG, e.getMessage(), e);
                          Toast.makeText(mContext,
                                 mContext.getString(R.string.insufficient_drm_rights),
                                 Toast.LENGTH_SHORT).show();
                      } catch (IOException e){
                          Log.e(TAG, e.getMessage(), e);
                          Toast.makeText(mContext,
                                 mContext.getString(R.string.insufficient_drm_rights),
                                 Toast.LENGTH_SHORT).show();
                      

                      }
                  }

mHandler.removeCallbacks(r);

mHandler.postr;

代码慢慢看,思路很简单:其实就是在postRunnable之前先清除队列中已存的相同Runnable实例。 这样可以保证同时队列中只有一个操作在处理 。


很遗憾,不生效。:(,改动之后,问题依然存在,欲哭无泪 。


再来,我将整个模式改为message再试试,核心代码如下 :

if(mHandler.hasMessages(MEDIA_PLAY_WHAT_MESSAGEFLAG))

{

return ;

}

Messagemsg = mHandler.obtainMessage() ;

msg.what= this.MEDIA_PLAY_WHAT_MESSAGEFLAG ;

msg.obj= mMeidaPlayMessageObj ;

mHandler.sendMessageDelayed(msg,1000) ;


代码慢慢看,思路也很简单,通过发消息的方式, 先检测如果有相关消息队列,就直接跳出函数,不做任何处理,否则延迟一秒后再向队列发送一条消息 。


为何我用了1秒这个这么长的时间呢,因为这么长时间如果都处理不了,那就不是压力测试的问题了,而是方法本身的问题了,这也是通过排除法来试图排除是因为点击屏幕过快产生的问题。

编译,再试 ,很不辛,又不生效,不幸被我猜中了 。 仰望苍天 !


现在问题很明显了:不是压力测试时候点击过快导致的ANR,而是某些方法本身有问题。

通过之前我们的日志

-----pid 2922 at 2011-01-13 13:51:07 -----
Cmdline: com.android.mms

DALVIKTHREADS:
"main"prio=5 tid=1 NATIVE
  |group="main" sCount=1 dsCount=0 s=N obj=0x4001d8d0self=0xccc8
  |sysTid=2922 nice=0 sched=0/0 cgrp=default handle=-1345017808
  |schedstat=( 3497492306 15312897923 10358 )
  atandroid.media.MediaPlayer._release(Native Method)
  atandroid.media.MediaPlayer.release(MediaPlayer.java:1206)
  atandroid.widget.VideoView.stopPlayback(VideoView.java:196)
  atcom.android.mms.ui.SlideView.stopVideo(SlideView.java:640)

很容易就知道了问题出在每次执行完了MediaPlayer.stop()方法调用之后会调用release()来释放播放器资源。 而这个方法中又死在了_release()方法上。 这是一个Native方法。

因此,真相大白 ,问题是在Framework层的MediaPlayer调用的Native方法_release()


adb logcat命令查看并过滤android输出log

cmd命令行中使用adb logcat命令查看android系统和应用的log,dos窗口按ctrl+c中断输出log记录。

logcat日志中的优先级/tag标记:

android输出的每一条日志都有一个标记和优先级与其关联。

优先级是下面的字符,顺序是从低到高:
V — 明细 verbose(最低优先级)
D — 调试 debug
I — 信息 info
W — 警告 warn
E — 错误 error
F — 严重错误 fatal
S — 无记载 silent

标记是一个简短的字符串,用于标识原始消息的来源。如下是一个日志输出的消息,优先级是“D”,标记是“PowerManagerService”:

D/PowerManagerService( 305): onSensorChanged: light value: 306.0
使用logcat命令查看android日志输出:

adb logcat     //显示全部日志
adb logcat > c:\test.log //将日志保存到文件test.log

根据tag标记和级别过滤日志输出:

仅输出标记为“ActivityManager”且优先级大于等于“Info”和标记为“PowerManagerService”并且优先级大于等于“Debug”的日志:

adb logcat ActivityManager:I PowerManagerService:D *:S

注:*:S用于设置所有标记的日志优先级为S,这样可以确保仅输出符合条件的日志。

adb logcat *:W   //显示所有优先级大于等于“warning”的日志

adb logcat -s PowerManagerService   //显示PowerManagerService的日志信息

Logcat命令列表:

-d 将日志显示在控制台后退出
-c 清理已存在的日志
-f <filename> 将日志输出到文件
adb logcat -f /sdcard/test.txt

-v <format>设置日志输入格式控制输出字段,默认的是brief格式

brief — 显示优先级/标记和原始进程的PID (默认格式)
process — 仅显示进程PID
tag — 仅显示优先级/标记
thread — 仅显示进程:线程和优先级/标记
raw — 显示原始的日志信息,没有其他的元数据字段
time — 显示日期,调用时间,优先级/标记,PID
long —显示所有的元数据字段并且用空行分隔消息内容

adb logcat -v thread   //使用 thread 输出格式
注意-v 选项中只能指定一种格式。

-b <buffer>加载一个可使用的日志缓冲区供查看,默认值是main。
radio — 查看包含在无线/电话相关的缓冲区消息
events — 查看事件相关的消息
main — 查看主缓冲区 (默认缓冲区)

adb logcat -b radio     //查看radio缓冲区


五, logcat命令详解

Android日志系统提供了记录和查看系统调试信息的功能。日志都是从各种软件和一些系统的缓冲区中记录下来的,缓冲区可以通过logcat命令来查看和使用.

在使用 logcat 之前,请确保手机的 USB调试模式 已经开启,可以通过" Setting->Application->Development-> USB debugging"来开启。
logcat本身是android的shell的一个命令,你可以通过“adb shell”进入shell后执行logcat命令,也可以通过"adb logcat"直接运行。
语法:
[adb] logcat [<option>] ... [<filter-spec>] ...
选项:
-b  <buffer> 指定要查看的日志缓冲区,可以是 system , events  , radio,main  . 默认值是 system和main  。
-c  清楚屏幕上的日志. 
-d  输出日志到屏幕上. 
-f  <filename> 指定输出日志信息的<filename> ,默认是stdout . 
-g  输出指定的日志缓冲区,输出后退出. 
-n  <count> 设置日志的最大数目<count> .,默认值是4,需要和 -r 选项一起使用。 
-r   <kbytes> 每<kbytes> 时输出日志,默认值为16,需要和-f 选项一起使用. 
-s  设置默认的过滤级别为silent. 
-v   <format> 设置日志输入格式,默认的是brief 格式,要知道更多的支持的格式,参看Controlling Log Output 
参数<filter-spec>
参数 <filter-spec> 用于对某类的tag的 日志输出进行过滤 。每一个输出的Android日志信息都有一个tag和它的优先级.
日志的标签是系统部件原始信息的一个简要的标志。这个tag就是Log.i,Log.d,Log.i,Log.w,Log.e,Log.wtf系列函数中的tag.
对于System.out.print系列函数所对于的tag,其实就是"System.out"
<filter-spec>以“tag:priority”的形式来对日志输出进行过滤的
优先级priority有以下几种,按照从低到高顺利排列如下:
— Verbose (lowest priority) 对应于Log.i()系列函数
D  — Debug 对应于Log.d()系列函数
I  — Info 对应于Log.i()系列函数
W  — Warning 对应于Log.w()系列函数
E  — Error 对应于Log.e()系列函数
F  — Fatal 对应于Log.wtf()系列函数
S  — Silent (highest priority, on which nothing s ever printed)
在运行logcat的时候在前两列的信息中你就可以看到 logcat 的标签列表和优先级别,它是这样标出的:<priority>/<tag> .
<filter-spec>中的优先级是指显示 该优先级即其以上优先级得日志 比如对于 robin:D 表示显示tag为 robin 的所有 Debug及其以上优先级 的日志。<filter-spec>只是的针对某类的tag的日志进行过滤,如果有多个针对同一个tag的过滤的话,以最后一个为准。另外对于tag,可以使用通配符。对于在tag中没有使用统配符的<filter-spec>,我称它为显式的日志过滤器;而对于在tag中使用统配符的,我称它它为隐式的日志过滤器。如果这两种过滤器有对着同一个tag的过滤的话,以显示的日志过滤器为准。如果是同一种的话(显式/隐式),以后一个为准。
因为<filter-spec>只是指明了对某一类tag应该进行如何过滤 它没说明的其他tag,将采用系统默认的方式(*:V),即全部输出 。我们可以通过选项" -s "来设置<filter-spec>中没有说明的 tag 来都不输出,相当于" *.S "
实例1
adb logcat -s robin:i
这样将显现tag为robin的Info及以上优先级的所有的日志。该命令等同于 adb logcat robin:i *:S
-b 选项
该选项用于指定要操作的日志缓冲区,可以是 system , events  , radio , main  .它们分别对应 /dev/log 文件夹下的 system , events  , radio , main 日志文件 。系统默认的是 system main  。该选项可以出现多次,以指定多个日志缓冲去。
比如:
adb logcat -b system -b main -b events -b radio -s robin:i
日志输出的开头几行说明了你当前查看的哪些日志缓冲区,比如上面的语句的前几行就是:
--------- beginning of /dev/log/radio
--------- beginning of /dev/log/events
--------- beginning of /dev/log/system
--------- beginning of /dev/log/main
其实“ adb logcat -s robin:i ”相当于“ adb logcat -b system -b main -s robin:i ”。我们的Log.i,Log.d,Log.i,Log.w,Log.e,Log.wtf系列函数及 System.out.print 系列函数以及 System.erro.print 系列都输出到了 main 缓冲区。因此我们一般用默认的就足够了。
events 缓冲区对应的日志文件 /system/etc/event-log-tags ,使用android.util.EventLog生成的日志就输出到该缓冲区。
android.database.sqlite.SQLiteDatabase logTimeStat() 函数就是使用 EventLog 来进行日志输出的
-c 选项
该选项用于清空你所指定的日志缓冲区。应该就是清除其对应的日志文件
-s 选项
该选项将把 tag 的默认过滤级别设置为 silent ,这样tag默认就不显示。系统把tag的默认过滤级别是设置为Verbose,这样其tag默认就是要显示的。
-f 选项
该选项指定输出日志信息的<filename> ,默认是stdout . 但是这里的文件是指android系统上的文件。如果我们想把日志输出到本地window系统的话,请采用如下形式的命令:
adb logcat -s robin:i>1.log
这样日志就输出了你的window的当前目录的1.log文件中。
-v 选项
日志信息包括了许多元数据域包括标签和优先级。可以通过 -v选项 可以用来指定 日志的输出格式,以显示出特定的元数据域。
brief  —  Display priority/tag and PID of originating process (the default format).显示prority/tag,产生日志的进程的id,和日志消息本身。它是日志默认的输出格式。
process  —  Display PID only.显示priority,产生日志的进程的id,和日志消息本身
tag  —  Display the priority/tag only.显示prority/tag,和消息本身
thread  —  Display process:thread and priority/tag only.显示priority,线程和日志消息本身
raw  —  Display the raw log message, with no other metadata fields.只显示消息本身
time  —  Display the date, invocation time, priority/tag, and PID of the originating process.显示产生日志的时间,prority/tag,产生日志的进程Id,和日志消息本身。
long  —  Display all metadata fields and separate messages with a blank lines.显示产生日志的时间,prority/tag,产生日志的进程Id,和日志消息本身。但是日志消息本身另其一行进行显示。每个日志之间空一行。
当启动了 logcat  ,你可以通过-v 选项来指定输出格式:
[adb] logcat [-v <format>]
实例2:
adb logcat  -v time  -s robin:v
注意是通过 -v 选项 来设置输出格式.

如何统计Activity启动所用的时间?

可以通过分析Log得到(这个就是DDMS的那个Log)。
当我们点击触摸时会了类似以下的Log  A
03-06 03:36:47.865: VERBOSE/InputDevice(2486): ID[0]=0(0)  Dn (0=>1)
03-06 03:36:47.865: INFO/PowerManagerService(2486): Ulight 3->7|0
03-06 03:36:47.933: INFO/(2486): PRODUCT_SHIP = false
03-06 03:36:47.933: VERBOSE/InputDevice(2486): ID[0]=0(0)  Up (1=>0)
当系统收到一个启动Activity的Intent时会有类似以下的Log  B
03-06 03:36:47.954:  INFO/ActivityManager(2486): Starting activity : Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.teleca/.ContextMenuActivity }
当系统完成Activity进程启动时会有类似以下的Log  C
03-06 03:36:48.068:  INFO/ActivityManager(2486): Start proc  com.teleca for activity com.teleca/.ContextMenuActivity: pid=2933 uid=10089 gids={}
当系统完成显示一个Activity时会有类似以下的Log D:
03-06 03:36:48.302:  INFO/ActivityManager(2486): Displayed activity com.teleca/.ContextMenuActivity: 240 ms (total 41289 ms)
那么启动Activity所用的 所有时间= Log D 那刻 时间  减去 Log  A  那刻 时间。
即: 03:36:48.302 减去  03:36:47.933,简化即为: 48.302 减去 47.933=369ms.
注1: Log A Log B 为系统所用时间。:
注2: Log BLog D主要 activity 本身创建所用的时间(on create(),onresume(),onDraw()等) 48.302减去 47.954=348ms.有时也包括了 系统为 创建 activity所在应用程序(APK) 进程所用的时间
注3: Log D   240 ms  (total 41289 ms) 的意义还不清楚。只是”240ms“和 Log C Log D的时间很接近。
附件1:
一个启动 activity的Log:
03-06 03:36:47.865: VERBOSE/InputDevice(2486): ID[0]=0(0) Dn(0=>1)
03-06 03:36:47.865: INFO/PowerManagerService(2486): Ulight 3->7|0
03-06 03:36:47.933: INFO/(2486): PRODUCT_SHIP = false
03-06 03:36:47.933: VERBOSE/InputDevice(2486): ID[0]=0(0) Up(1=>0)
03-06 03:36:47.935: VERBOSE/WindowManager(2486): Dsptch 1 x82.0 y592.0 > Window{485bd6c0 com.sec.android.app.twlauncher/com.sec.android.app.twlauncher.Launcher paused=false}
03-06 03:36:47.954: INFO/ActivityManager(2486): Starting activity: Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.teleca/.ContextMenuActivity }
03-06 03:36:48.029: INFO/Launcher(2570): onPause()
03-06 03:36:48.044: DEBUG/AudioHardwareALSA(2377):  ALSA OPEN mode 0,device 2 
03-06 03:36:48.044: INFO/AudioHardwareALSA(2377): Try to open ALSA PLAYBACK device AndroidPlayback_Speaker_normal
03-06 03:36:48.048: WARN/Launcher(2570): WallpaperManager setVisibility visible true
03-06 03:36:48.068: INFO/ActivityManager(2486): Start proc com.teleca for activity com.teleca/.ContextMenuActivity: pid=2933 uid=10089 gids={}
03-06 03:36:48.099: INFO/AudioHardwareALSA(2377): Initialized ALSA PLAYBACK device AndroidPlayback_Speaker_normal
03-06 03:36:48.099: DEBUG/AudioHardwareALSA(2377): Set PLAYBACK PCM format to S16_LE (Signed 16 bit Little Endian)
03-06 03:36:48.099: DEBUG/AudioHardwareALSA(2377): Using 2 channels for PLAYBACK.
03-06 03:36:48.099: DEBUG/AudioHardwareALSA(2377): Set PLAYBACK sample rate to 44100 HZ
03-06 03:36:48.099: DEBUG/AudioHardwareALSA(2377): Buffer size: 2048
03-06 03:36:48.099: DEBUG/AudioHardwareALSA(2377): Latency: 46439
03-06 03:36:48.134: INFO/WindowManager(2486): Setting rotation to 1, animFlags=1
03-06 03:36:48.154: INFO/ActivityManager(2486): Config changed: { scale=1.0 imsi=0/0 loc=en_CA touch=3 keys=1/1/2 nav=1/1 orien=2 layout=34 uiMode=17 seq=3 FlipFont=0}
03-06 03:36:48.189: INFO/Launcher(2570): onWindowFocusChanged(false)
03-06 03:36:48.302: INFO/ActivityManager(2486): Displayed activity com.teleca/.ContextMenuActivity: 240 ms (total 41289 ms)


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值