目前有2種策略可用,第一個和線程相關,它主要針對主線程(或UI線程)。由於在主線程中讀寫磁盤和進行網絡訪問都不是好的做法,Google已經在磁盤和網絡代碼中添加了嚴苛模式(StrictMode)鉤子(hook)。如果你對某個線程打開嚴苛模式(StrictMode),當那個線程進行磁盤和網絡訪問,你將獲得警告。你可以選擇警告方式。一些違例包含用戶慢速調用(custom slow calls 這麼翻譯行嗎?),磁盤讀寫,網絡訪問。你能選擇將警告寫入LogCat,顯示一個對話框,閃下屏幕,寫入DropBox日誌文件,或讓應用崩潰。最通常的做法是寫入LogCat或讓應用崩潰。
-
寫程序時,你應該始終假定下列兩種情況:
網絡很慢(你正在試圖連接的服務器甚至可能沒有響應);
文件系統的訪問速度很慢。
結論就是,不應該在主線程內進行網絡操作或訪問文件系統,因爲緩慢的操作會拖累系統的響應能力。雖然在開發環境中,你可能永遠不會遇到任何網絡問題或任何文件系統的性能問題,但用戶可能不像你那麼幸運。
注意 SD卡並不都具有相同“速度”,如果應用在很大程度上依賴外部存儲設備的性能,那麼你應該確保在來自不同製造商的各種SD卡上測試過你的應用。
Android有實用工具來幫助應用檢測這類缺陷。它提供的StrictMode是檢測不良行爲的良好工具。通常情況下,在應用啓動時,即當onCreate()被調用時,啓用StrictMode,如代碼清單1-15所示。
在應用中啓用StrictMode
[java] view plaincopy- public class MyApplication extends Application {
- @Override
- public void onCreate () {
- super.onCreate();
- StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()
- .detectCustomSlowCalls()// API等級11,使用StrictMode.noteSlowCode
- .detectDiskReads()
- .detectDiskWrites()
- .detectNetwork()
- .penaltyLog()
- .penaltyFlashScreen()// API等級11
- .build());
- // 其實和性能無關,但如果使用StrictMode,最好也定義VM策略
- StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()
- .detectLeakedSqlLiteObjects()
- .detectLeakedClosableObjects()// API等級11
- .setClassInstanceLimit(Class.forName("com.apress.proandroid.SomeClass", 100)// API等級11
- .penaltyLog()
- .build());
- }
- }
StrictMode是Android 2.3引入的,在Android 3.0中加入了更多的功能,所以應該確保選擇了正確的Android版本,讓代碼跑在適當的Android平臺上,如代碼清單1-12所示。
Android 3.0中引入的需要特別留意的方法包括detectCustomSlowCall()和noteSlowCall(),它們都是用來檢測應用中執行緩慢的代碼或潛在緩慢的代碼。代碼清單1-16說明了如何將代碼標記爲潛在緩慢的代碼。
-
-
代碼清單1-16 標記潛在緩慢的代碼
[java] view plaincopy- public class Fibonacci {
- public static BigInteger computeRecursivelyWithCache(int n) {
- StrictMode.noteSlowCall("computeRecursivelyWithCache");// 消息可以帶任何信息
- SparseArray cache = new SparseArray();
- return computeRecursivelyWithCache(n, cache);
- }
- ...
- }
-
[java] view plaincopy[java] view plaincopy
- public class Fibonacci {
- public static BigInteger computeRecursivelyWithCache(int n) {
- StrictMode.noteSlowCall("computeRecursivelyWithCache");// 消息可以帶任何信息
- SparseArray cache = new SparseArray();
- return computeRecursivelyWithCache(n, cache);
- }
- ...
- }
從主線程調用computeRecursivelyWithCache執行時間過長,如果StrictMode Thread 策略配置爲檢測緩慢調用時,會出現如下日誌:
[java] view plaincopy- StrictMode policy violation; ~duration=21121 ms:
- android.os.StrictMode$StrictModeCustomViolation: policy=31 violation=8 msg=computeRecursivelyWithCache
StrictMode policy violation; ~duration=21121 ms:
android.os.StrictMode$StrictModeCustomViolation: policy=31 violation=8 msg=computeRecursivelyWithCacheAndroid提供了一些輔助方法,可以在主線程裏進行臨時磁盤讀寫,如代碼
[java] view plaincopy- StrictMode.ThreadPolicy oldPolicy = StrictMode.allowThreadDiskReads();
修改線程策略,臨時允許磁盤讀取
[java] view plaincopy- StrictMode.ThreadPolicy oldPolicy = StrictMode.allowThreadDiskReads();
- // 從磁盤讀取數據
- StrictMode.setThreadPolicy(oldPolicy);
目前沒有臨時允許網絡訪問的方法,但實在沒有理由在主線程中允許這種訪問,即使是暫時的,也沒有合適的方式知道訪問是否很快。有人可能會說,也沒有合理的方式知道磁盤訪問將是否是快速的,但那是另一場爭論。
注意 只在開發階段啓用StrictMode,發佈應用時,記得要禁用它。如果你使用detectAll()方法去建立策略總是可行的,那將來更可行,未來的Android版本會檢測出更多的不良行爲。