转载请注明地址:http://blog.csdn.NET/yincheng886337/article/details/50524709

StrictMode(严格模式)使用

StrictMode严格模式,主要用来检测程序中违例情况的开发者工具。最常用的场景就是检测主线程中本地磁盘、网络读写等耗时的操作以及Activity泄露等,但该模式不建议在Release版本开启,此外该模式无法监控JNI中的磁盘IO和网络请求且其违例情况仅供参考,需结合实际开发需求予以解决。

StrictMode检测什么?

主要采用采用ThreadPolicy(线程策略)和VmPolicy(Vm策略)进行检测,各策略检测内容如下:

ThreadPolicy

线程策略检测的内容有

· 自定义的耗时调用 使用 detectCustomSlowCalls() 开启

· 磁盘读取操作 使用 detectDiskReads() 开启

· 磁盘写入操作 使用 detectDiskWrites() 开启

· 网络操作 使用 detectNetwork() 开启

VmPolicy

虚拟机策略检测的内容有

· Activity泄露 使用 detectActivityLeaks() 开启

· 未关闭的Closable对象泄露 使用 detectLeakedClosableObjects() 开启

· 泄露的Sqlite对象 使用 detectLeakedSqlLiteObjects() 开启

· 检测实例数量 使用 setClassInstanceLimit() 开启

StrictMode具体使用

public class DebugUtil {
public static void startStrictModeVmPolicy(){
StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()
.detectActivityLeaks()/*检测Activity内存泄露*/
.detectLeakedClosableObjects()/*检测未关闭的Closable对象*/
.detectLeakedSqlLiteObjects() /*检测Sqlite对象是否关闭*/
/*也可以采用detectAll()来检测所有想检测的东西*/
.penaltyLog().build());
}
public static void startStrictModeThreadPolicy(){
StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()
.detectDiskReads()/*磁盘读取操作检测*/
.detectDiskWrites()/*检测磁盘写入操作*/
.detectNetwork() /*检测网络操作*/
/*也可以采用detectAll()来检测所有想检测的东西*/
.penaltyLog().build());
}
}

如果你想检测整个App或某Activity的相关泄露问题,可在Application或Activity的onCreate方法中直接调用DebugUtil中封装好的策略即可

如何查看检测结果?

只需要查看TAG为StrictMode的日志即可,如:logcat -c;logcat -s StrictMode 或者adb logcat | grep StrictMode

如何解决检测出来的问题?

针对此问题下面给出几点建议,仅供参考:

1.如果是主线程中出现文件读写违例问题,建议使用工作线程(可采用HandlerThread,IntentService、线程池或直接new Thread,必要时可结合Handler)完成,但采用工作线程在某个Activity中操作时注意线程要能正常结束,否则将导致内存泄露,相关细节后文将有表述。

2.如果是对SharedPrefrences写入操作,在API 9以上建议优先调用apply而非commit,此外需注意的是确保SharedPrefrences在单进程中使用,如果涉及跨进程数据交换,建议自己编写跨进程SharedPrefrences实现机制,否则可能导致数据不准确。

3.如果存在未关闭的Closable对象,需根据对应的stacktrace进行关闭。

4.如果SQLite对象泄露,根据对应的stacktrace进行释放。

5.注意registerBroadcast和unregisterBroadcast配对使用,对文件操作完成后记得close操作。

StrictMode使用示例

现以主线程中文件读写为例,引起违例警告的代码如下:

public void writeToExternalStorage() {
File externalStorage = Environment.getExternalStorageDirectory();
File destFile = new File(externalStorage, "dest.txt");
try {
OutputStream output = new FileOutputStream(destFile, true);
output.write("droidyue.com".getBytes());
output.flush();
output.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}

StrictMode违例警告:

D/StrictMode( ): StrictMode policy violation; ~duration= ms: android.os.StrictMode$StrictModeDiskReadViolation: policy= violation=
D/StrictMode( ): at android.os.StrictMode$AndroidBlockGuardPolicy.onReadFromDisk(StrictMode.java:)
D/StrictMode( ): at libcore.io.BlockGuardOs.open(BlockGuardOs.java:)
D/StrictMode( ): at libcore.io.IoBridge.open(IoBridge.java:)
D/StrictMode( ): at java.io.FileOutputStream.<init>(FileOutputStream.java:)
D/StrictMode( ): at com.example.strictmodedemo.MainActivity.writeToExternalStorage(MainActivity.java:)
D/StrictMode( ): at com.example.strictmodedemo.MainActivity.onCreate(MainActivity.java:)
D/StrictMode( ): at android.app.Activity.performCreate(Activity.java:)

解决方法:

public void writeToExternalStorage() {
new Thread(){
@Override
public void run() {
/*将对读写操作移至线程*/
File externalStorage = Environment.getExternalStorageDirectory();
File destFile = new File(externalStorage, "dest.txt");
OutputStream output = null;
try {
output = new FileOutputStream(destFile, true);
output.write("droidyue.com".getBytes());
output.flush();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}finally{
/*对文件操作完成后,注意关闭*/
if(output != null){
output.close();
}
}
}
}
}

Android内存泄露分析之StrictMode的更多相关文章

  1. JVM内存管理概述与android内存泄露分析

    一.内存划分 将内存划分为六大部分,分别是PC寄存器.JAVA虚拟机栈.JAVA堆.方法区.运行时常量池以及本地方法栈. 1.PC寄存器(线程独有):全称是程序计数寄存器,它记载着每一个线程当前运行的 ...

  2. (转)专项:Android 内存泄露实践分析

    今天看到一篇关于Android 内存泄露实践分析的文章,感觉不错,讲的还算详细,mark到这里. 原文发表于:Testerhome: 作者:ycwdaaaa ;  原文链接:https://teste ...

  3. Android Studio 使用Memory Monitor进行内存泄露分析

    在使用Android Studio进行内存泄露分析之前,我们先回顾一下Java相关的内存管理机制,然后再讲述一下内存分析工具如何使用. 一.Java内存管理机制 1. Java内存分配策略 Java ...

  4. Android内存机制分析1——了解Android堆和栈

    //----------------------------------------------------------------------------------- Android内存机制分析1 ...

  5. Android 内存泄漏分析与解决方法

    在分析Android内存泄漏之前,先了解一下JAVA的一些知识 1. JAVA中的对象的创建 使用new指令生成对象时,堆内存将会为此开辟一份空间存放该对象 垃圾回收器回收非存活的对象,并释放对应的内 ...

  6. Android内存机制分析2——分析APP内存使用情况

    上面一篇文章说了Android应用运行在dalvik里面分配的堆和栈内存区别,以及程序中什么代码会在哪里运行.今天主要是讲解一下Android里面如何分析我们程序内存使用情况.以便后续可以分析我们程序 ...

  7. Android内存泄露调试

    Android 内存泄漏调试 一.概述 如果我们编写的代码当中有太多的对内存使用不当的地方,难免会使得我们的设备运行缓慢,甚至是死机.为了能够使得 Android 应用程序安全且快速的运行, Andr ...

  8. 关于内存泄露分析插件 MAT 的用法

    关于内存泄露分析插件 MAT 的用法,建议大家有时间看一下,下面的文章 http://www.blogjava.net/rosen/archive/2010/05/21/321575.html htt ...

  9. Android 内存管理分析(四)

    尊重原创作者,转载请注明出处: http://blog.csdn.net/gemmem/article/details/8920039 最近在网上看了不少Android内存管理方面的博文,但是文章大多 ...

随机推荐

  1. 在Ubuntu14.04中配置mysql远程连接教程

    上一篇文章,小编带大家学会了在Ubuntu14.04中安装MySQL,没有来得及上课的小伙伴们可以戳这篇文章:如何在Ubuntu14.04中安装mysql,今天给大家分享一下,如何简单的配置MySQL ...

  2. PHP设置30秒内对页面的访问次数

    <?php //Calculate 60 days in the future //seconds * minutes * hours * days + current time $intime ...

  3. Mysql学习总结(11)——MySql存储过程与函数

    摘要:存储过程和函数是在数据库中定义一些SQL语句的集合,然后直接调用这些存储过程和函数来执行已经定义好的SQL语句.存储过程和函数可以避免开发人员重复的编写相同的SQL语句.而且,存储过程和函数是在 ...

  4. 多线程模式之Master-Worker

    一. 介绍 需要使用Master-Worker的场景:主线程开了多个子进程(Worker进程)去执行任务时,主线程希望能收集到每个子进程的执行结果. 所以,Master-Worker模式基本上就是: ...

  5. POJ 2352 Stars(线段树)

    题目地址:id=2352">POJ 2352 今天的周赛被虐了. . TAT..线段树太渣了..得好好补补了(尽管是从昨天才開始学的..不能算补...) 这题还是非常easy的..维护 ...

  6. android 动态设置TextView值,例:金额添加

    一说到动态递增设置TextView值,非常多人应该立即就想到起个线程,让后在线程中睡眠指定时间,使用handler发送消息更新TextView值! 这样是实现了动态递增设置TextView值可是效率不 ...

  7. 第8章7节《MonkeyRunner源代码剖析》MonkeyRunner启动执行过程-小结

    最后我们对MonkeyRunner启动的过程做一个总结,当然,当中也包括启动Monkey,尽管它不属于启动过程的一部分: monkeyrunner这个shell脚本会先设置一些执行环境的系统属性保存到 ...

  8. Day1下午解题报告

    预计分数:0+30+30=60 实际分数:0+30+40=70 T1水题(water) 贪心,按长度排序, 对于第一幅牌里面的,在第二个里面,找一个长度小于,高度最接近的牌 进行覆盖. 考场上的我离正 ...

  9. 转 C#:使用MD5对用户密码加密与解密

    C#中常涉及到对用户密码的加密于解密的算法,其中使用MD5加密是最常见的的实现方式.本文总结了通用的算法并结合了自己的一点小经验,分享给大家. 一.使用16位.32位.64位MD5方法对用户名加密 1 ...

  10. Python爬虫之『urlopen』

    本文以爬取百度首页为示例来学习,python版本为python3.6.7,完整代码会在文章末附上 本次学习所用到的python框架:urllib.request 本次学习所用到的函数: urllib. ...