前言
Strict Mode翻译中文就是严格模式,是安卓提供的一套策略检测机制,测试一旦发现应用违反了这套策略,就会有提示产生。
一、应用开启使用Strict Mode
一般建议在Application.onCreate之后调用下方的代码,当然你可以任何时候开启这个功能,也可以添加或者删除你要检测的策略或者提示方式。
1.1 打开功能
public class App extends Application {
@Override
public void onCreate() {
super.onCreate();
//开启Thread策略模式
StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder().detectNetwork()//监测主线程使用网络io
.detectCustomSlowCalls()//监测自定义运行缓慢函数
.detectDiskReads() // 检测在UI线程读磁盘操作
.detectDiskWrites() // 检测在UI线程写磁盘操作
.penaltyLog() //写入日志
.penaltyDialog()//弹出弹框
.build());
//开启VM策略模式
StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder().detectLeakedSqlLiteObjects()//监测sqlite泄露
.detectLeakedClosableObjects()//监测没有关闭IO对象
.setClassInstanceLimit(MainActivity.class, 1) // 设置某个类的同时处于内存中的实例上限,可以协助检查内存泄露
.detectActivityLeaks()
.penaltyLog()//写入日志
.penaltyDeath()//出现上述情况异常终止
.build());
}
}
1.2 看看效果
我们写一个demo来违反一下策略
1.2.1 主线程中操作一下文件
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
private Button mTxtMain;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mTxtMain = (Button) findViewById(R.id.txt_main);
mTxtMain.setOnClickListener(this);
}
@Override
public void onClick(View v) {
try {
File file = new File(this.getFilesDir() + "/" + "1.txt");
if(!file.exists()) {
file.createNewFile();
}
} catch (Exception e) {
}
}
}
1.2.2 出现了弹出框
当我点击button的时候在主线程访问文件,就会有截图中的弹框出现。
二、开发者选项中的Strict Mode enabled
细心的朋友应该会发现开发者选项中有一个开关是Strict Mode enabled,这个开关有什么用,一开始我以为可以整体控制所有应用Strict Mode的开关,但是我发现我关闭之后,Demo还是会弹出框,然后我跟了一下源码。
2.1 代码分析
2.1.1 VISUAL_PROPERTY
开发者选项中开关会影响下面这个值
public static final String VISUAL_PROPERTY = "persist.sys.strictmode.visual";
2.1.2 initThreadDefaults
initThreadDefaults是在attachApplicationLocked之后被调用,也就是所有应用都会走下面这个代码,并且这个代码比Application onCreate执行的早。所以这就是所有应用的一个通用策略,App可以按照自己的需求,override策略,就是第一节里的Demo。
public static void initThreadDefaults(ApplicationInfo ai) {
final ThreadPolicy.Builder builder = new ThreadPolicy.Builder();
final int targetSdkVersion =
(ai != null) ? ai.targetSdkVersion : Build.VERSION_CODES.CUR_DEVELOPMENT;
// Build.VERSION_CODES.HONEYCOMB版本之后不允许主线程进行网络请求
if (targetSdkVersion >= Build.VERSION_CODES.HONEYCOMB) {
builder.detectNetwork();
builder.penaltyDeathOnNetwork();
}
if (Build.IS_USER || DISABLE || SystemProperties.getBoolean(DISABLE_PROPERTY, false)) {
//如果手机是user版本或者disable,什么都不做
} else if (Build.IS_USERDEBUG) {
//如果手机是USERDEBUG版本,并且App是BundledSystemApp,开启所有策略检查
if (isBundledSystemApp(ai)) {
builder.detectAll();
builder.penaltyDropBox();
//如果开发者选项开启了Strict Mode enabled,检查到app违反规则的时候
//就会在屏幕周围有一圈红色的闪一下。
if (SystemProperties.getBoolean(VISUAL_PROPERTY, false)) {
builder.penaltyFlashScreen();
}
}
} else if (Build.IS_ENG) {
//如果手机是ENG版本,并且App是BundledSystemApp,开启所有策略检查
if (isBundledSystemApp(ai)) {
builder.detectAll();
builder.penaltyDropBox();
builder.penaltyLog();
builder.penaltyFlashScreen();//检查到app违反规则的时候,就会在屏幕周围有一圈红色的闪一下。
}
}
setThreadPolicy(builder.build());
}
2.1.3 isBundledSystemApp
什么样子的APP才是BundledSystemApp
system_sever进程
系统App,并且包名带有android的字样(phone进程,google play,chrome浏览器除外)
public static boolean isBundledSystemApp(ApplicationInfo ai) {
if (ai == null || ai.packageName == null) {
// Probably system server
return true;
} else if (ai.isSystemApp()) {
// Ignore unbundled apps living in the wrong namespace
if (ai.packageName.equals("com.android.vending")
|| ai.packageName.equals("com.android.chrome")) {
return false;
}
// Ignore bundled apps that are way too spammy
// STOPSHIP: burn this list down to zero
if (ai.packageName.equals("com.android.phone")) {
return false;
}
if (ai.packageName.equals("android")
|| ai.packageName.startsWith("android.")
|| ai.packageName.startsWith("com.android.")) {
return true;
}
}
return false;
}
2.2 小结
开发者选项中的Strict Mode enabled只是控制在userdebug版本的固件上,BundledSystemApp应用违反Strict Mode策略的时候,是否显示红色的警示框而已。
三、总结
Strict Mode并不是强制的,开发人员可以按照需求来控制,如果大家在开发应用的早期开启Strict Mode,可以提升App的质量,试着在你目前开发的App中开启一下Strict Mode,可能你会发现到处都是警示弹框。
网友评论