StrictMode简介

StrictMode最常用来捕捉应用程序的主线程,它将报告与线程及虚拟机相关的策略违例。一旦检测到策略违例policy violation,你将获得警告,其包含了一个栈trace显示你的应用在何处发生违例。除了主线程,我们还可以在Handler,AsyncTask,AsyncQueryHandler,IntentService等API中使用StrictMode。

示例代码  StrictMode文档


检查策略

StrictMode的线程策略主要用于检测磁盘IO和网络访问,而虚拟机策略主要用于检测内存泄漏现象。Android已经在磁盘IO访问和网络访问的代码中加入了StrictMode。当监视的线程发生策略的违例时,就可以获得警告,例如写入LogCat,显示一个对话框,闪下屏幕,写入DropBox日志文件,或让应用崩溃。最通常的做法是写入LogCat或让应用崩溃。下面的代码展示了如何使用StrictMode的检查策略:

public void onCreate() {
if (DEVELOPER_MODE) {
StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder() //线程策略
.detectDiskReads() //detect [dɪˈtɛkt] vt.查明,发现; 洞察; 侦察,侦查;
.detectDiskWrites()
.detectNetwork()
// or .detectAll() for all detectable problems
.penaltyDialog() //penalty [ˈpɛnəlti] n.刑罚; 惩罚; 害处
.penaltyLog()
.penaltyDropBox() //DropBox下拉框
.build()); StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder() //虚拟机策略
.detectLeakedSqlLiteObjects()
.detectLeakedClosableObjects()
.penaltyLog()
.penaltyDeath()
.build());
}
super.onCreate();
}
21
21
 
1
public void onCreate() {  
2
     if (DEVELOPER_MODE) {  
3
         StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()  //线程策略
4
                 .detectDiskReads()  //detect [dɪˈtɛkt] vt.查明,发现; 洞察; 侦察,侦查;
5
                 .detectDiskWrites()  
6
                 .detectNetwork()   
7
                 // or .detectAll() for all detectable problems 
8
                 .penaltyDialog()  //penalty [ˈpɛnəlti] n.刑罚; 惩罚; 害处
9
                 .penaltyLog()  
10
                 .penaltyDropBox()  //DropBox下拉框
11
                 .build());  
12
         
13
         StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()  //虚拟机策略
14
                 .detectLeakedSqlLiteObjects()  
15
                 .detectLeakedClosableObjects()  
16
                 .penaltyLog()  
17
                 .penaltyDeath()  
18
                 .build());  
19
     }  
20
     super.onCreate();  
21
 }  

使用方法

如果不指定检测函数,也可以用detectAll()来替代。penaltyLog()表示将警告输出到LogCat,你也可以使用其他或增加新的惩罚(penalty)函数,例如使用penaltyDeath()的话,一旦StrictMode消息被写到LogCat后应用就会崩溃。具体支持的监视方法见:StrictMode.ThreadPolicy.Builder 与 StrictMode.VmPolicy.Builder

在正式版本中,我们并不希望使用StrictMode来让用户的应用因为一个警告而崩溃,所以在应用正式发布时,需要移出这些监视。你可以通过删除代码来实现,不过这里提供一个更好的方式来解决这个问题,即使用AndroidMainifest文件中的debuggable属性来实现,代码如下所示:

android:debuggable="true"  
1
1
 
1
android:debuggable="true"  

PS:可能在gradle中这么配置也行

android {
buildTypes {
debug {
debuggable true
}
}
}
7
7
 
1
android {    
2
    buildTypes {
3
        debug {
4
            debuggable true
5
        }
6
    }
7
}

在代码中,使用方法如下所示:

// Return if this application is not in debug mode
if ((context.getApplicationInfo().flags & ApplicationInfo.FLAG_DEBUGGABLE) != 0) {
// Do StrictMode setup here
StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()
.detectLeakedSqlLiteObjects()
.penaltyLog()
.penaltyDeath()
.build());
}
9
9
 
1
// Return if this application is not in debug mode
2
if ((context.getApplicationInfo().flags & ApplicationInfo.FLAG_DEBUGGABLE) != 0) {   
3
    // Do StrictMode setup here   
4
    StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()   
5
        .detectLeakedSqlLiteObjects()   
6
        .penaltyLog()   
7
        .penaltyDeath()   
8
        .build());   
9
}  

实例

我们在测试代码的主线程中去访问网络,这样就一定会触发StrictMode的线程监测,代码如下所示:

public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()
.detectDiskReads()
.detectDiskWrites()
.detectNetwork()
.penaltyDialog()
.penaltyLog()
.penaltyDropBox()
.build());
super.onCreate(savedInstanceState); TextView mTextView = new TextView(this);
setContentView(mTextView); InputStream inputStream = null;
try {
HttpResponse httpResponse = new DefaultHttpClient().execute(new HttpGet("http://www.baidu.com"));
HttpEntity httpEntity = httpResponse.getEntity();
if (httpResponse.getStatusLine().getStatusCode() == 200) {
inputStream = httpEntity.getContent();
mTextView.setText(new BufferedReader(new InputStreamReader(inputStream)).readLine());
}
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (inputStream != null) inputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
35
35
 
1
public class MainActivity extends Activity {
2
    @Override
3
    protected void onCreate(Bundle savedInstanceState) {
4
        StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()
5
                .detectDiskReads()
6
                .detectDiskWrites()
7
                .detectNetwork()
8
                .penaltyDialog()
9
                .penaltyLog()
10
                .penaltyDropBox()
11
                .build());
12
        super.onCreate(savedInstanceState);
13
        
14
        TextView mTextView = new TextView(this);
15
        setContentView(mTextView);
16
        
17
        InputStream inputStream = null;
18
        try {
19
            HttpResponse httpResponse = new DefaultHttpClient().execute(new HttpGet("http://www.baidu.com"));
20
            HttpEntity httpEntity = httpResponse.getEntity();
21
            if (httpResponse.getStatusLine().getStatusCode() == 200) {
22
                inputStream = httpEntity.getContent();
23
                mTextView.setText(new BufferedReader(new InputStreamReader(inputStream)).readLine());
24
            }
25
        } catch (Exception e) {
26
            e.printStackTrace();
27
        } finally {
28
            try {
29
                if (inputStream != null) inputStream.close();
30
            } catch (IOException e) {
31
                e.printStackTrace();
32
            }
33
        }
34
    }
35
}

运行代码,并将Log信息保存到本地,在Log中,我们可以搜索D/StrictMode关键字,如下所示:

D/StrictMode: StrictMode policy violation; ~duration=5131 ms: android.os.StrictMode$StrictModeDiskReadViolation: policy=183 violation=2
at android.os.StrictMode$AndroidBlockGuardPolicy.onReadFromDisk(StrictMode.java:1157)
at libcore.io.BlockGuardOs.open(BlockGuardOs.java:182)
at libcore.io.IoBridge.open(IoBridge.java:480)
at java.io.FileInputStream.<init>(FileInputStream.java:76)
at java.io.FileInputStream.<init>(FileInputStream.java:103)
at android.content.res.HwResources.readDefaultConfig(HwResources.java:1172)
at android.content.res.HwResources.loadDrawable(HwResources.java:574)
at android.content.res.Resources.getDrawable(Resources.java:809)
at android.content.Context.getDrawable(Context.java:403)
at com.android.internal.widget.ToolbarWidgetWrapper.setIcon(ToolbarWidgetWrapper.java:321)
at com.android.internal.widget.ActionBarOverlayLayout.setIcon(ActionBarOverlayLayout.java:741)
at com.android.internal.policy.impl.PhoneWindow.setDefaultIcon(PhoneWindow.java:1661)
at android.app.Activity.initWindowDecorActionBar(Activity.java:2174)
at android.app.Activity.setContentView(Activity.java:2189)
at com.imooc.strictmodetest.MainActivity.onCreate(MainActivity.java:36)
at android.app.Activity.performCreate(Activity.java:6102)
at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1106)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2403)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2512)
at android.app.ActivityThread.access$1200(ActivityThread.java:165)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1373)
at android.os.Handler.dispatchMessage(Handler.java:102)
at android.os.Looper.loop(Looper.java:135)
at android.app.ActivityThread.main(ActivityThread.java:5593)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:967)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:762)
29
29
 
1
D/StrictMode: StrictMode policy violation; ~duration=5131 ms: android.os.StrictMode$StrictModeDiskReadViolation: policy=183 violation=2
2
      at android.os.StrictMode$AndroidBlockGuardPolicy.onReadFromDisk(StrictMode.java:1157)
3
      at libcore.io.BlockGuardOs.open(BlockGuardOs.java:182)
4
      at libcore.io.IoBridge.open(IoBridge.java:480)
5
      at java.io.FileInputStream.<init>(FileInputStream.java:76)
6
      at java.io.FileInputStream.<init>(FileInputStream.java:103)
7
      at android.content.res.HwResources.readDefaultConfig(HwResources.java:1172)
8
      at android.content.res.HwResources.loadDrawable(HwResources.java:574)
9
      at android.content.res.Resources.getDrawable(Resources.java:809)
10
      at android.content.Context.getDrawable(Context.java:403)
11
      at com.android.internal.widget.ToolbarWidgetWrapper.setIcon(ToolbarWidgetWrapper.java:321)
12
      at com.android.internal.widget.ActionBarOverlayLayout.setIcon(ActionBarOverlayLayout.java:741)
13
      at com.android.internal.policy.impl.PhoneWindow.setDefaultIcon(PhoneWindow.java:1661)
14
      at android.app.Activity.initWindowDecorActionBar(Activity.java:2174)
15
      at android.app.Activity.setContentView(Activity.java:2189)
16
      at com.imooc.strictmodetest.MainActivity.onCreate(MainActivity.java:36)
17
      at android.app.Activity.performCreate(Activity.java:6102)
18
      at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1106)
19
      at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2403)
20
      at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2512)
21
      at android.app.ActivityThread.access$1200(ActivityThread.java:165)
22
      at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1373)
23
      at android.os.Handler.dispatchMessage(Handler.java:102)
24
      at android.os.Looper.loop(Looper.java:135)
25
      at android.app.ActivityThread.main(ActivityThread.java:5593)
26
      at java.lang.reflect.Method.invoke(Native Method)
27
      at java.lang.reflect.Method.invoke(Method.java:372)
28
      at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:967)
29
      at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:762)

主要看第一行

D/StrictMode: StrictMode policy violation; ~duration=5131 ms: android.os.StrictMode$StrictModeDiskReadViolation: policy=183 violation=2
1
1
 
1
D/StrictMode: StrictMode policy violation; ~duration=5131 ms: android.os.StrictMode$StrictModeDiskReadViolation: policy=183 violation=2

这行Log中可以显示出StrictMode提示的原因,通过这里的TraceLog我们就可以来找到优化的方法。

除了在Logcat中查看StrictMode的日志信息,如果你使用了penaltyDropbox()方法,那么你还可以通过如下所示的命令来调用DropBoxManager观察StrictMode日志:

adb shell dumpsys dropbox data_app_strictmode --print
1
1
 
1
adb shell dumpsys dropbox data_app_strictmode --print

同时,如果使用了penaltyDialog()方法,在应用中还会弹出如下所示的提示框:


暂停监测

如果在程序运行中无法避免的会违反StrictMode中的一些定义好的策略,而我们又希望能够暂时忽略这些策略的监视,我们可以使用permitXXXXX方法来暂停这些内容的监测,在做完需要忽略的监测之后,再启用监测,代码如下所示:
StrictMode.ThreadPolicy old = StrictMode.getThreadPolicy();
StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder(old)
.permitDiskWrites()
.build());
//doSomethingWriteToDisk();
StrictMode.setThreadPolicy(old);
x
 
1
StrictMode.ThreadPolicy old = StrictMode.getThreadPolicy();  
2
StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder(old)  
3
        .permitDiskWrites()  
4
        .build());  
5
//doSomethingWriteToDisk();  
6
StrictMode.setThreadPolicy(old);  

2017-10-20

App优化 StrictMode 严格模式的更多相关文章

  1. 4.Android App 优化之消除卡顿

    转载:http://gold.xitu.io/post/582583328ac247004f3ab124 1, 感知卡顿 用户对卡顿的感知, 主要来源于界面的刷新. 而界面的性能主要是依赖于设备的UI ...

  2. 斗牛app上架应用宝、牛牛手机游戏推广、百人牛牛app应用开发、棋牌游戏上传、手游APP优化

    联系QQ:305-710439斗牛app上架应用宝.牛牛手机游戏推广.百人牛牛app应用开发.棋牌游戏上传.手游APP优化 iOS开发iPhone/iPad平台安卓手机软件开发机型覆盖范围 超过113 ...

  3. Android App优化之ANR详解

    引言 背景:Android App优化, 要怎么做? Android App优化之性能分析工具 Android App优化之提升你的App启动速度之理论基础 Android App优化之提升你的App ...

  4. 0428数字口袋精灵app优化

    "数字口袋精灵app"优化 目录: 一.项目github总仓库推送 二.开发成员 三.分工与合作 四.各模块成果 五.团队成员贡献分 内容: 一.项目github总仓库: http ...

  5. [实践] Android5.1.1源码 - 让某个APP以解释执行模式运行

    [实践] Android5.1.1源码 - 让某个APP以解释执行模式运行   作者:寻禹@阿里聚安全 前言 本文的实践修改了Android5.1.1的源码. 本文只简单的讲了一下原理.在“实践”一节 ...

  6. App开发三种模式

    APP开发三种模式 现在App开发的模式包含以下三种: Native App 原生开发AppWeb App 网页AppHybrid App 混合原生和Web技术开发的App 详细介绍: http:// ...

  7. 【高德地图API】那些年我们一起开发的APP—即LBS应用模式分享

    原文:[高德地图API]那些年我们一起开发的APP—即LBS应用模式分享 摘要:利用地图API都能做些什么应用呢?应用商店里所有的分类,都可以结合上LBS来丰富应用.除了传统的生活服务应用,还有新潮的 ...

  8. Android App优化建议(转载)

    假如要Google Play上做一个最失败的案例,那最好的秘诀就是界面奇慢无比.耗电.耗内存.接下来就会得到用户的消极评论,最后名声也就臭了.即使你的应用设计精良.创意无限也没用. 耗电或者内存占用等 ...

  9. App优化 Systrace

    简介 trace [tres] vt.跟踪,追踪; 追溯,探索; 探索; 查找; n.痕迹; 痕迹,踪迹; 微量,极少量; 1 1   1 trace [tres] vt.跟踪,追踪; 追溯,探索; ...

随机推荐

  1. CSUOJ 1087 就多了两分钟

    Description Yucept21和他的室友Zyn因为宿舍没电去网吧上网,上了27分钟,Cs打电话来说来电了.所以Yucept21在第29分钟下机了,上网的费用是一块钱,然后Zyn墨迹了两分钟, ...

  2. 二、django rest_framework源码之认证流程剖析

    1 绪言 上一篇中讲了django rest_framework总体流程,整个流程中最关键的一步就是执行dispatch方法.在dispatch方法中,在调用了一个initial方法,所有的认证.权限 ...

  3. 从Table 表中取出第 m 条到第 n 条的记录

    * FROM Table id FROM Table )) --从TABLE表中取出第m到n条记录 (Exists版本) * FROM TABLE AS a WHERE Not Exists ( * ...

  4. Web2.0应用程序的7条原则

    个人看好Web的发展潜力,本文字摘自<Collective Intelligence 实战> 网络是平台 使用传统许可模式软件的公司或用户必须运行软件.定期更新至最新版本,以及扩展它来满足 ...

  5. 2018haoi总结

    AM T1 写了40分,有50分的暴力分,只看到n=1能用费马小定理,没看到还有一个mod质数也能用费马小定理的10分. AM T2 写了10分,60分异或方程求自由元. AM T3 现在还没搞清楚题 ...

  6. LOJ P3960 列队 树状数组 vector

    https://www.luogu.org/problemnew/show/P3960 树状数组预处理之后直接搞就可以了,也不是很好解释,反正就是一个模拟过程的暴力用树状数组维护,还挺巧妙的. 我为什 ...

  7. NOIP2018 RP++

    飞吧,不用看向地面. NOIP,RP++.

  8. pear中几个实用的xml代码库

    1.XML_Beautifier 用于将一段排版凌乱的XML文档美化 <?php require_once "XML/Beautifier.php"; $fmt = new ...

  9. Codechef December Challenge 2014 Chef and Apple Trees 水题

    Chef and Apple Trees Chef loves to prepare delicious dishes. This time, Chef has decided to prepare ...

  10. Mysql show indexes 查看索引状态

     查看表中有哪些已建立的索引 SHOW INDEX FROM tbl_name [FROM db_name] SHOW INDEX会返回以下字段: | Table | Non_unique | Key ...