StrictMode的使用_leewokan的博客-程序员ITS301_strictmode 忽略某个

最新的Android平台中(Android 2.3起),新增加了一个新的类,叫StrictMode(android.os.StrictMode)。这个类可以用来帮助开发者改进他们编写的应用,并且提供了各种的策略,这些策略能随时检查和报告开发者开发应用中存在的问题,比如可以监视那些本不应该在主线程中完成的工作或者其他的一些不规范和不好的代码。

  StrictMode有多种不同的策略,每一种策略又有不同的规则,当开发者违背某个规则时,每个策略都有不同的方法去显示提醒用户。在本文中,将举例子说明如何使用在Android 中使用 StrictMode。

  StrictMode的策略和规则

  目前,有两大类的策略可供使用,一类是关于常用的监控方面的,另外一类是关于VM虚拟机等方面的策略。常用的监控方面的策略有如下这些:

  Disk Reads 磁盘读

  Disk Writes 磁盘写

  Network access 网络访问

  Custom Slow Code 自定义的运行速度慢的代码分析

  前面三种的意思读者应该很清楚,就是正如它们的名字所示,分别对磁盘的读和写,网络访问进行监控。而第四种的自定义慢代码分析,是仅当访问调用类的时后才触发的,可以通过这种方法去监视运行缓慢的代码。当在主线程中调用时,这些验证规则就会起作用去检查你的代码。比如,当你的应用在下载或者解析大量的数据时,你可以触发自定义运行速度慢代码的查询分析,作用很大。StrictMode可以用于捕捉发生在应用程序主线程中耗时的磁盘、网络访问或函数调用,可以帮助开发者使其改进程序,使主线程处理UI和动画在磁盘读写和网络操作时变得更平滑,避免主线程被阻塞的发生。

  而VM方面的策略重点关注如下几类:

  内存泄露的Activity对象

  内存泄露的SQLite对象

  内存泄露的释放的对象

  其中,内存泄露的Activity对象和内存泄露的SQLite对象都比较好理解,而所谓对关闭对象的检查,主要是去监那些本该释放的对象,比如应该调用close()方法的对象。

  当开发者违反某类规则时,每种策略都会有不同的方法令开发者知道当时的情况。相关的违反情况可以记录在LogCat中或者存储在DropBox中(android.os.DropBox)服务中。而常用监控类的策略还会在当违规情况发生时显示相关的对话框和当时的上下文环境,所有的这些都为了能让开发者尽快地了解程序的瑕疵,以提交程序的质量。下面分步讲解如何使用stritctmode。

 


  第一步 启用strictmode

  为了能在应用中启用和配置StrictMode,开发者最好尽可能在应用程序的生命周期的早段使用,方法是调用StrictMode的方法setThreadPolicy。当使用常用监控类的时候,一个最好的调用时机,是在应用中入口和activities被调用前进行。比如在一个应用程序中,可以把代码放在启动Activity类的onCreate()方法中,下面是一个代码示例,启用了当前情况下的所有策略及规则,当程序中出现违背常用的规则时,将会显示相关的提示信息窗口:

   StrictMode.setThreadPolicy( new StrictMode.ThreadPolicy.Builder()
        .detectAll()
        .penaltyLog()
        .penaltyDialog()
打印logcat,当然也可以定位到dropbox,通过文件保存相应的log
        .build());
StrictMode.setVmPolicy(
new StrictMode.VmPolicy.Builder().detectAll()
        .penaltyLog()
        .build());

  当然,以上代码只应在未发布上线的测试版本的应用中运行以方便监视相关的运行情况,当在生产版本上时不应该启用strictmode。因此,最佳的代码实践应该为如下的样子:

   public void onCreate() {  
    
if (DEVELOPER_MODE) {  
        StrictMode.setThreadPolicy(
new StrictMode.ThreadPolicy.Builder()  
                .detectDiskReads()  
                .detectDiskWrites()  
                .detectNetwork()  
                .penaltyLog()  
                .build());  
    }  
    super.onCreate();  
}  

  第二步 运行strictmode

  当应用启用了strictmode模式时,其实跟普通的应用没什么两样,在测试和运行时,跟平时运行普通应用程序一样就可以了。当启用了Strictmode模式时,会监视所有的程序运行情况,当发现出现重大问题或违背策略规则时,会提示用户。下面是当运行启用了strictmode模式的应用时,当发现违背规则时,显示给用户的信息,细心观察下跟普通的出错信息有什么不同吧。

   09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ): StrictMode policy violation; ~duration = 319 ms: android.os.StrictMode$StrictModeDiskWriteViolation: policy = 31 violation = 1
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.os.StrictMode$AndroidBlockGuardPolicy.onWriteToDisk(StrictMode.java: 1041 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.database.sqlite.SQLiteStatement.acquireAndLock(SQLiteStatement.java: 219 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.database.sqlite.SQLiteStatement.executeUpdateDelete(SQLiteStatement.java: 83 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.database.sqlite.SQLiteDatabase.updateWithOnConflict(SQLiteDatabase.java: 1829 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.database.sqlite.SQLiteDatabase.update(SQLiteDatabase.java: 1780 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at com.mamlambo.tutorial.tutlist.data.TutListProvider.update(TutListProvider.java: 188 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.content.ContentProvider$Transport.update(ContentProvider.java: 233 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.content.ContentResolver.update(ContentResolver.java: 847 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at com.mamlambo.tutorial.tutlist.data.TutListProvider.markItemRead(TutListProvider.java: 229 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at com.mamlambo.tutorial.tutlist.TutListFragment.onListItemClick(TutListFragment.java: 99 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.support.v4.app.ListFragment$ 2 .onItemClick(ListFragment.java: 53 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.widget.AdapterView.performItemClick(AdapterView.java: 282 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.widget.AbsListView.performItemClick(AbsListView.java: 1037 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.widget.AbsListView$PerformClick.run(AbsListView.java: 2449 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.widget.AbsListView$ 1 .run(AbsListView.java: 3073 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.os.Handler.handleCallback(Handler.java: 587 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.os.Handler.dispatchMessage(Handler.java: 92 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.os.Looper.loop(Looper.java: 132 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at android.app.ActivityThread.main(ActivityThread.java: 4123 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at java.lang.reflect.Method.invokeNative(Native Method)
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at java.lang.reflect.Method.invoke(Method.java: 491 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java: 841 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java: 599 )
09 - 04 16 : 15 : 34.592 : DEBUG / StrictMode( 15883 ):     at dalvik.system.NativeStart.main(Native Method

  并且会出现如下的提示窗口,提示用户:

Android最佳实践之:StrictMode介绍

  忽略某些规则

  应该说大部分由StrictMode产生的规则警示都应去遵守,但有时也不是所有产生的信息都表明你的程序有错误。比如,在应用程序的主线程中去快速读写磁盘其实不会对应用的性能产生太大的影响,又或者你在调试程序阶段有一些调试的代码违反了设定的规则,这些都可以忽略掉这些规则。

  忽略规则有两种方法,一种是单纯在代码中把Strictmode的代码注释掉,另外一种比较好的方法是,在需要忽略的时候和地方,增加相应的代码去让系统停止使用这些规则去检查,等开发者认为有必要检查时,再重新应用这些规则,比如:

  StrictMode.ThreadPolicy old = StrictMode.getThreadPolicy();

  StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder(old)

  .permitDiskWrites()

  .build());

  doCorrectStuffThatWritesToDisk();

  StrictMode.setThreadPolicy(old);

  这里首先用old来保存了当前的策略规则,然后doCorrectStuffThatWritesToDisk();

  这里,执行了一些向磁盘快速读写的操作,最后又重新启用了这些规则。

  小结

  StrictMode是一个十分有用的类,它可以很方便地应用于检查Android应用程序的性能和存在的问题。当开启这个模式后,开发者能很好地检查应用中存在的潜在问题,更多的请参考Android文档中的相关API说明。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/leewokan/article/details/8649570

智能推荐

.NET高级工程师面试经历_喜欢猪猪的博客-程序员ITS301

1.SQL Server数据库操作的原子性,出Select之外,Update、Insert、Delete的操作都是原子性的,不可拆分,执行的最小单位;可以用于充值交费中 ,如果多个请求进行更新同一条 数据时,直接使用update Table1 set money=money+100 这种方式就可以避免多个语句,更新一条记录导致的更新失败的问题(一般想法是,先查询当前的账户余额,然后进行更新,这...

转:Latex数学宏包集锦_jueshu的博客-程序员ITS301_mathbf宏包

<div style="width: 656px; margin: 0; padding: 0; height: 0;"></div> <center> algorithm2e它提供一个 Algorithm2e 环境,...

解决 java.util.Iterator java.util.List.iterator()' on a null object reference 异常的办法_willwaywang6的博客-程序员ITS301

前言最近,负责解决线上捕获的 bug,其中一个就是 java.util.Iterator java.util.List.iterator()’ on a null object reference 问题。很明显这样的异常是由于遍历了一个为 null 的集合导致的。难道我们仅仅对集合进行判空处理就可以了吗?正文先说一下我自己的代码情况,代码示例如下:...

Ubuntu中快速打开终端_青青草原我最狂~的博客-程序员ITS301

方法一:(推荐) 1.Ctrl+Alt+T:打开终端 / 以新窗口的形式打开终端 2.Ctrl+Shift+T:在终端中以新选项页的形式打开终端 3.Alt+数字键:快速跳转终端 4.Ctrl+D:快速关闭终端方法二: 1.Ctrl+F2:打开命令输入框 2.输入“gnome-termin...

EXT7 textcell与gridcell区别_路过君_P的博客-程序员ITS301_grid和cell

textcell用于显示纯文本内容,不会从column继承renderer、tpl和formattergridcell是grid的默认cell类型源码Ext.define('Ext.grid.cell.Cell', { extend: 'Ext.grid.cell.Text', xtype: 'gridcell', ... // 与column关联时继承ren...

通过net.sf.json.xml.XMLSerializer转换xml为json后格式不对的问题_梦忆柯南的博客-程序员ITS301_net.sf.json.xml.xmlserializer

可能是因为xml字符串中多加了或减少了空格照成的。 比如12,字符串间没有多余空格,会解析成对象, 如果12 这样中间多了一个空格,就可能解析成数组...

随便推点

深入研究Clang(四) Clang编译器的简单分析_snsn1984的博客-程序员ITS301

首先我们确定下Clang编译器的具体内容和涵盖范围。之前在《LLVM每日谈之二十 Everything && Clang driver 》中曾经提到过,Clang driver(命令行表示是clang)和Clang前端(按照具体实现来说就是Clang的那些库所实现的前端)是不同的,同时还存在一个Clang编译器(命令行表示是clang -cc1)。Clang编译器不仅仅包含了Clan

java 按钮不填充填充 panel,Java Swing - 如何在JScrollPane中创建用Buttons填充的JPanel_Fayyy Li的博客-程序员ITS301

import java.awt.Dimension;import java.awt.FlowLayout;import java.awt.GridLayout;import javax.swing.JButton;import javax.swing.JFrame;import javax.swing.JPanel;import javax.swing.JScrollPane;public cla...

Java并发编程之接口ExecutorService的方法使用_迷糊弟弟的博客-程序员ITS301_executorserviceadapter

7.1 在ThreadPoolExecutor中使用ExecutorService中的方法方法invokeAny()和invokeAll()具有阻塞特性。方法invokeAny()取得第一个完成任务的结果值,当第一个任务执行完成后,会调用interrupt()方法将其他任务中断,所以在这些任务中可以结合if(Thread.currentThread().isInterrupted()==tru...

Angular4中调用js代码_Mingyueyixi的博客-程序员ITS301_angular 使用js

Angular4中调用js代码 引言:Angular2开始采用TypeScript进行编码。本文讲述如何在Angular4项目中调用原生的 js 代码。本文代码: http://git.oschina.net/mingyueyixi/angular4-usejs代码使用环境:webstrom 调用过程可以归纳为:引入 js 文件到项目中编写 TypeScript 声明文件在 .angu

计算机会计知识竞赛,全国企业会计信息化知识竞赛试题及答案_weixin_39813263的博客-程序员ITS301

全国企业会计信息化知识竞赛试题及答案第 1 题(单选)下列各项中,符合会计软件不可逆的记账功能要求的是( A )A.记账标记不得通过任何操作予以取消B.不可以用红字凭证更正记账错误C.可以删除已经记账的记账凭证D.可以进行反记账第 2 题(单选)下列关于信息系统自动生成会计凭证的表述中,正确的是( C )A.信息系统自动生成的所有会计凭证,都需要先经过人工审核再进行后续处理B.信息系统自动生成的会...

MFC 模仿编译器属性 设置控件_志汉天涯的博客-程序员ITS301

在很多软件中,都可以看到属性设置,比如 VS 的属性设置,本控件就是这样的功能,希望对需要的人提供一些参考。可以在外面套入一个DockPane,称为真正可移动的属性栏。需要的,我后面可以放上来。

推荐文章

热门文章

相关标签