嚴(yán)苛模式(StrictMode)

Android 2.3提供一個(gè)稱為嚴(yán)苛模式(StrictMode)的調(diào)試特性,Google稱該特性已經(jīng)使數(shù)百個(gè)Android上的Google應(yīng)用程序受益。那它都做什么呢?它將報(bào)告與線程及虛擬機(jī)相關(guān)的策略違例。一旦檢測(cè)到策略違例(policy violation),你將獲得警告,其包含了一個(gè)棧trace顯示你的應(yīng)用在何處發(fā)生違例。你可以強(qiáng)制用警告代替崩潰(crash),也可以僅將警告計(jì)入日志,讓你的應(yīng)用繼續(xù)執(zhí)行。策略的細(xì)節(jié)尚難確定,可以期待隨Android的成熟Google將增加更多策略。

目前有2種策略可用,第一個(gè)和線程相關(guān),它主要針對(duì)主線程(或UI線程)。由于在主線程中讀寫(xiě)磁盤(pán)和進(jìn)行網(wǎng)絡(luò)訪問(wèn)都不是好的做法,Google已經(jīng)在磁盤(pán)和網(wǎng)絡(luò)代碼中添加了嚴(yán)苛模式(StrictMode)鉤子(hook)。如果你對(duì)某個(gè)線程打開(kāi)嚴(yán)苛模式(StrictMode),當(dāng)那個(gè)線程進(jìn)行磁盤(pán)和網(wǎng)絡(luò)訪問(wèn),你將獲得警告。你可以選擇警告方式。一些違例包含用戶慢速調(diào)用(custom slow calls 這么翻譯行嗎?),磁盤(pán)讀寫(xiě),網(wǎng)絡(luò)訪問(wèn)。你能選擇將警告寫(xiě)入LogCat,顯示一個(gè)對(duì)話框,閃下屏幕,寫(xiě)入DropBox日志文件,或讓?xiě)?yīng)用崩潰。最通常的做法是寫(xiě)入LogCat或讓?xiě)?yīng)用崩潰。列表2-9顯示了一個(gè)為線程策略設(shè)置嚴(yán)苛模式(StrictMode)的例子。

列表2-9 設(shè)置嚴(yán)苛模式(StrictMode)的線程策略

view plainprint?

StrictMode.setThreadPolicy(newStrictMode.ThreadPolicy.Builder()

.detectDiskReads()

.detectDiskWrites()

.detectNetwork()

.penaltyLog()

.build());

Builder類使得設(shè)置變得很簡(jiǎn)單,Builder函數(shù)定義所有策略都返回Builder對(duì)象,從而這些函數(shù)能像列表2-9那樣串連在一起。最后調(diào)用build()函數(shù)返回一個(gè)ThreadPolicy對(duì)象作為StrictMode對(duì)象的setThreadPolicy()函數(shù)的參數(shù)。注意到setThreadPolicy()是一個(gè)靜態(tài)函數(shù),因此不需要實(shí)例化StrictMode對(duì)象。在內(nèi)部,setThreadPolicy()將對(duì)當(dāng)前線程應(yīng)用該策略。如果不指定檢測(cè)函數(shù),也可以用detectAll()來(lái)替代。penaltyLog()表示將警告輸出到LogCat,你也可以使用其他或增加新的懲罰(penalty)函數(shù),例如使用penaltyDeath()的話,一旦StrictMode消息被寫(xiě)到LogCat后應(yīng)用就會(huì)崩潰。

你不需要頻繁打開(kāi)嚴(yán)苛模式(StrictMode),你可以在主活動(dòng)的onCreate()函數(shù)中打開(kāi)它,你也可以在Application派生類的OnCreate()函數(shù)中設(shè)置嚴(yán)苛模式(StrictMode)。線程中運(yùn)行的任何代碼都可以設(shè)置嚴(yán)苛模式(StrictMode),但你的確只需要設(shè)置一次,一次就夠了。

類似于線程策略(ThreadPolicy),嚴(yán)苛模式(StrictMode)有虛擬機(jī)策略(VmPolicy)。虛擬機(jī)策略(VmPolicy)能檢查內(nèi)存泄漏,譬如,當(dāng)關(guān)閉一個(gè)SQLite對(duì)象前的完結(jié)操作,或其他任何類似可關(guān)閉對(duì)象在關(guān)閉前的完結(jié)操作。虛擬機(jī)策略(VmPolicy)由一個(gè)類似的Builder類創(chuàng)建,如列表2-10所示。和線程策略(ThreadPolicy)不同的是,虛擬機(jī)策略(VmPolicy)不能通過(guò)一個(gè)對(duì)話框提供警告。

列表2-10 設(shè)置嚴(yán)苛模式(StrictMode)的虛擬機(jī)策略

view plainprint?

StrictMode.setVmPolicy(newStrictMode.VmPolicy.Builder()

.detectLeakedSqlLiteObjects()

.penaltyLog()

.penaltyDeath()

.build());

因?yàn)樵O(shè)置發(fā)生在線程中,嚴(yán)苛模式(StrictMode)甚至能在從一個(gè)對(duì)象到另一個(gè)對(duì)象的控制流中找到違例事件。當(dāng)違例發(fā)生,你會(huì)驚奇地注意到代碼正運(yùn)行于主線程,而棧trace將幫助你發(fā)現(xiàn)它如何發(fā)生。于是你能單步調(diào)試解決問(wèn)題,或是將代碼移到它自己的后臺(tái)線程,或是就保持原來(lái)的處理方式。這都取決與你。當(dāng)然,你可能希望適時(shí)關(guān)閉嚴(yán)苛模式(StrictMode),當(dāng)你的程序作為產(chǎn)品發(fā)布時(shí),你可不希望它僅為了一個(gè)警告在你的用戶手里崩潰。

有兩個(gè)方法可以關(guān)閉嚴(yán)苛模式(StrictMode),最直接的就是移除相應(yīng)代碼,但這樣做不利于持續(xù)開(kāi)發(fā)的產(chǎn)品。你通常可以定義一個(gè)應(yīng)用級(jí)別布爾變量來(lái)測(cè)試是否需要調(diào)用嚴(yán)苛模式(StrictMode)代碼。在發(fā)布產(chǎn)品前將這個(gè)值定義為FALSE。更優(yōu)雅的方式是利用調(diào)試模式(debug mode)的特點(diǎn),在AndroidManifest.xml中定義這個(gè)布爾變量。字段的屬性之一是android:debuggable,其義自明。列表2-11給出了利用該特性的控釋方法。

列表2-11 僅在調(diào)試模式設(shè)置嚴(yán)苛模式(StrictMode)

view plainprint?

//?Return?if?this?application?is?not?in?debug?mode

ApplicationInfo?appInfo?=?context.getApplicationInfo();

intappFlags?=?appInfo.flags;

if((appFlags?&?ApplicationInfo.FLAG_DEBUGGABLE)?!=0)?{

//?Do?StrictMode?setup?here

StrictMode.setVmPolicy(newStrictMode.VmPolicy.Builder()

.detectLeakedSqlLiteObjects()

.penaltyLog()

.penaltyDeath()

.build());

}

使用Eclipse調(diào)試環(huán)境,ADT自動(dòng)為你設(shè)置debuggable屬性,使項(xiàng)目更易于管理。當(dāng)你在模擬器上或直接在設(shè)備上部署應(yīng)用,debuggable屬性為T(mén)RUE,當(dāng)你導(dǎo)出應(yīng)用建立一個(gè)產(chǎn)品版本,ADT將該屬性置為FALSE。注意,如果你另行設(shè)置了這個(gè)屬性值,ADT不會(huì)改變它。

嚴(yán)苛模式(StrictMode)很不錯(cuò),不過(guò)在Android 2.3之前的版本上該模式不工作。為了避免這個(gè)問(wèn)題,你要在StrictMode對(duì)象還不存在的時(shí)候就驗(yàn)證版本是否在Android2.3及以上。你能利用反射技術(shù)(reflection),當(dāng)嚴(yán)苛模式(StrictMode)函數(shù)有效時(shí)間接調(diào)用它,反之不去調(diào)用。方法很簡(jiǎn)單,你能按列表2-12中的代碼處理

列表2-12? 利用反射技術(shù)(reflection)調(diào)用嚴(yán)苛模式(StrictMode)

view plainprint?

try{

Class?sMode?=?Class.forName("android.os.StrictMode");

Method?enableDefaults?=?sMode.getMethod("enableDefaults");

enableDefaults.invoke(null);

}

catch(Exception?e)?{

//?StrictMode?not?supported?on?this?device,?punt

Log.v("StrictMode","...?not?supported.?Skipping...");

}

當(dāng)嚴(yán)苛模式(StrictMode)不存在,將捕捉到ClassNotFoundException異常。enableDefault()是嚴(yán)苛模式(StrictMode)類的另一個(gè)函數(shù),它檢測(cè)所有違例并寫(xiě)入LogCat。因?yàn)檫@里調(diào)用的是靜態(tài)形式的enableDefault(),所以用null作為參數(shù)傳入。

某些時(shí)候你不希望報(bào)告所有違例。那在主線程之外的其他線程中設(shè)置嚴(yán)苛模式(StrictMode)很不錯(cuò)。譬如,你需要在正在監(jiān)視的線程中進(jìn)行磁盤(pán)讀取。此時(shí),你要么不去調(diào)用detectDiskReads(),要么在調(diào)用detectAll()之后跟一個(gè)permitDiskReads()。類似允許函數(shù)也適用于其他操作。但要是你要在Anroid2.3之前版本上做這些事,有辦法嗎?當(dāng)然有。

當(dāng)應(yīng)用中嚴(yán)苛模式(StrictMode)無(wú)效,如果你試圖訪問(wèn)它,將拋出一個(gè)VerifyError異常。如果你將嚴(yán)苛模式(StrictMode)封裝在一個(gè)類里,并捕捉這個(gè)錯(cuò)誤,當(dāng)嚴(yán)苛模式(StrictMode)無(wú)效時(shí),你能忽略它。列表2-13顯示一個(gè)簡(jiǎn)單的嚴(yán)苛模式(StrictMode)封裝類StrictModeWrapper。列表2-14顯示了如何在你的應(yīng)用中使用這個(gè)封裝類。

列表 2–13 在Anroid2.3之前版本建立嚴(yán)苛模式(StrictMode)封裝類

view plainprint?

importandroid.content.Context;

importandroid.content.pm.ApplicationInfo;

importandroid.os.StrictMode;

publicclassStrictModeWrapper?{

publicstaticvoidinit(Context?context)?{

//?check?if?android:debuggable?is?set?to?true

intappFlags?=?context.getApplicationInfo().flags;

if((appFlags?&?ApplicationInfo.FLAG_DEBUGGABLE)?!=0)?{

StrictMode.setThreadPolicy(newStrictMode.ThreadPolicy.Builder()

.detectDiskReads()

.detectDiskWrites()

.detectNetwork()

.penaltyLog()

.build());

StrictMode.setVmPolicy(newStrictMode.VmPolicy.Builder()

.detectLeakedSqlLiteObjects()

.penaltyLog()

.penaltyDeath()

.build());

}

}

}

列表 2–14? 在Anroid2.3之前版本調(diào)用嚴(yán)苛模式(StrictMode)封裝類

view plainprint?

try{

StrictModeWrapper.init(this);

}

catch(Throwable?throwable)?{

Log.v("StrictMode","...?is?not?available.?Punting...");

}

//如果考慮到關(guān)于版本兼容問(wèn)題,因?yàn)榘凑丈厦娴膶?xiě)法在2.3以下系統(tǒng)是沒(méi)有問(wèn)題的,但是在2.3以上的話,就會(huì)出錯(cuò),所以應(yīng)該采用以下方式來(lái)處理:

[java]view plaincopy

@SuppressLint("NewApi")

publicstaticvoidinit(Context?context)?{

//?check?if?android:debuggable?is?set?to?true

intappFlags?=?context.getApplicationInfo().flags;

if((appFlags?&?ApplicationInfo.FLAG_DEBUGGABLE)?!=0)?{

try{

//Android?2.3及以上調(diào)用嚴(yán)苛模式

Class?sMode?=?Class.forName("android.os.StrictMode");

Method?enableDefaults?=?sMode.getMethod("enableDefaults");

enableDefaults.invoke(null);

}catch(Exception?e)?{

//?StrictMode?not?supported?on?this?device,?punt

Log.v("StrictMode","...?not?supported.?Skipping...");

}

/*

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

*?.detectDiskReads()?.detectDiskWrites()?.detectNetwork()

*?.penaltyLog()?.build());?StrictMode.setVmPolicy(new

*?StrictMode.VmPolicy.Builder()?.detectLeakedSqlLiteObjects()

*?.penaltyLog()?.penaltyDeath()?.build());

*/

}

}


http://blog.csdn.net/brokge/article/details/8543145

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,983評(píng)論 6 537
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,772評(píng)論 3 422
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 176,947評(píng)論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 63,201評(píng)論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,960評(píng)論 6 410
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 55,350評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,406評(píng)論 3 444
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,549評(píng)論 0 289
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,104評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,914評(píng)論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,089評(píng)論 1 371
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,647評(píng)論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,340評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,753評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 36,007評(píng)論 1 289
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,834評(píng)論 3 395
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,106評(píng)論 2 375

推薦閱讀更多精彩內(nèi)容

  • *面試心聲:其實(shí)這些題本人都沒(méi)怎么背,但是在上海 兩周半 面了大約10家 收到差不多3個(gè)offer,總結(jié)起來(lái)就是把...
    Dove_iOS閱讀 27,196評(píng)論 30 471
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語(yǔ)法,類相關(guān)的語(yǔ)法,內(nèi)部類的語(yǔ)法,繼承相關(guān)的語(yǔ)法,異常的語(yǔ)法,線程的語(yǔ)...
    子非魚(yú)_t_閱讀 31,726評(píng)論 18 399
  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,799評(píng)論 18 139
  • 對(duì)象的創(chuàng)建與銷毀 Item 1: 使用static工廠方法,而不是構(gòu)造函數(shù)創(chuàng)建對(duì)象:僅僅是創(chuàng)建對(duì)象的方法,并非Fa...
    孫小磊閱讀 2,015評(píng)論 0 3
  • 當(dāng)你自主創(chuàng)業(yè)時(shí),有必要先進(jìn)行這個(gè)測(cè)試,它可以幫助你判斷你自己是否適合創(chuàng)業(yè)?當(dāng)然,這個(gè)測(cè)試結(jié)果,業(yè)只供參考,因?yàn)闆Q定...
    破天船長(zhǎng)閱讀 662評(píng)論 0 2