性能优化工具(一)-StrictMode

一、简介:

StrictMode类是Android 2.3 (API 9)引入的一个工具类,可以用来帮助开发者发现代码中的一些不规范的问题,以达到提升应用响应能力的目的。举个例子来说,如果开发者在UI线程中进行了网络操作或者文件系统的操作,而这些缓慢的操作会严重影响应用的响应能力,甚至出现ANR对话框。为了在开发中发现这些容易忽略的问题,我们使用StrictMode,系统检测出主线程违例的情况并做出相应的反应,最终帮助开发者优化和改善代码逻辑。

二、官网文档:StrictMode使用文档
三、用途:

使用严格模式,系统检测出主线程违例的情况会做出相应的反应,如日志打印,弹出对话框亦或者崩溃等。换言之,严格模式会用不同的策略对应用的违例细节做监控,通过一定的方式暴露给开发者方便优化与改善。

四、使用:

通常在Activity或者自定义的Application类的onCreate方法中启动StrictMode(自定义application无效,不知道是不是配置问题)

// DEVELOPER_MODE代表只在调试模式下启用StrictMode

if (DEVELOPER_MODE) {

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

           .detectDiskReads()

           .detectDiskWrites()

           .detectNetwork()   // or .detectAll() for all detectable problems

           .penaltyLog()

           .build());

   StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()

           .detectLeakedSqlLiteObjects()

           .detectLeakedClosableObjects()

           .penaltyLog()

           .penaltyDeath()

           .build());

}

StrictMode通过策略方式来让你自定义需要检查哪些方面的问题,主要有两种策略:

ThreadPolicy:线程方面的策略

VmPolicy:VM方面的策略

ThreadPolicy策略:

StrictMode.ThreadPolicy.Builder主要方法如下:

    detectNetwork():用于检查UI线程中是否有网络请求

    detectDiskReads()和detectDiskWrites():是磁盘读写检查

    detectCustomSlowCalls():主要用于帮助开发者发现UI线程中调用的哪些方法执行的比较慢,要和StrictMode.noteSlowCall配合使用,只有通过  StrictMode.noteSlowCall标记“可能会”执行比较慢的方法,只有标记过的方法才能被检测到,日志中会记录方法的执行时间(注意:只有在主线程中执行的方法才会显示执行时间,在其他线程中执行的方法,就算是使用StrictMode.noteSlowCall标记,在日志中也不会打印执行时间)

VMPloicy策略:

StrictMode.VmPolicy.Builder()主要方法如下:

detectActivityLeaks() 用户检查 Activity 的内存泄露情况

detectLeakedClosableObjects() 和 detectLeakedSqlLiteObjects(),资源没有正确关闭时回触发,detectLeakedSqlLiteObjects() 和 detectLeakedClosableObjects()的用法类似,只不过是用来检查 SQLiteCursor 或者 其他 SQLite 对象是否被正确关闭

detectLeakedRegistrationObjects() 用来检查 BroadcastReceiver 或者 ServiceConnection 注册类对象是否被正确释放

setClassInstanceLimit(),设置某个类的同时处于内存中的实例上限,可以协助检查内存泄露

惩罚:

penaltyDeath():当触发违规条件时,直接Crash掉当前应用程序

penaltyDeathOnNetwork():当触发网络违规时,Crash掉当前应用程序

penaltyDialog():触发违规时,显示对违规信息对话框

penaltyFlashScreen():会造成屏幕闪烁,不过一般的设备可能没有这个功能

penaltyDropBox():将违规信息记录到 dropbox 系统日志目录中(/data/system/dropbox),你可以通过如下命令进行插件

adb shell dumpsys dropbox dataappstrictmode --print

permitCustomSlowCalls()、permitDiskReads ()、permitDiskWrites()、permitNetwork: 如果你想关闭某一项检测,可以使用对应的permit*方法

最后:我们也可以在Android设备的设置(Settings)中启用StrictMode(亲测没有什么效果):

开发者选项 > 启用严格模式

最后分享一个小例子:

public class MainActivity extends AppCompatActivity{
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()
                .detectAll()
                .penaltyLog()
                .build());

        findViewById(R.id.btn).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                writeToExternalStorage();
            }
        });
    }

    /**
     * 文件系统的操作
     */
    public void writeToExternalStorage() {
            try {
                File externalStorage = Environment.getExternalStorageDirectory();

                File mbFile = new File(externalStorage, "xxx.txt");
                if (mbFile.exists()){
                    mbFile.createNewFile();
                }
            OutputStream output = new FileOutputStream(mbFile, true);
            output.write("www.wooyun.org".getBytes());
            output.flush();
            output.close();
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

使用penaltyLog():

D/StrictMode: StrictMode policy violation; ~duration=3 ms: android.os.StrictMode$StrictModeDiskWriteViolation: policy=65567 violation=1
                                                                         at android.os.StrictMode$AndroidBlockGuardPolicy.onWriteToDisk(StrictMode.java:1253)
                                                                         at java.io.FileOutputStream.(FileOutputStream.java:220)
                                                                         at com.perf.strictmodedemo.MainActivity.writeToExternalStorage(MainActivity.java:44)
                                                                         at com.perf.strictmodedemo.MainActivity$1.onClick(MainActivity.java:28)
                                                                         at android.view.View.performClick(View.java:5619)
                                                                         at android.view.View$PerformClick.run(View.java:22298)
                                                                         at android.os.Handler.handleCallback(Handler.java:754)
                                                                         at android.os.Handler.dispatchMessage(Handler.java:95)
                                                                         at android.os.Looper.loop(Looper.java:165)
                                                                         at android.app.ActivityThread.main(ActivityThread.java:6365)
                                                                         at java.lang.reflect.Method.invoke(Native Method)
                                                                         at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:883)
                                                                         at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:773)

使用penaltyDialog():

性能优化工具(一)-StrictMode_第1张图片

笔者使用后感觉,StrictMode适合作为一些基础规范问题排查。

你可能感兴趣的:(性能优化工具(一)-StrictMode)