詳解Windows Phone開發(fā)中的墓碑機制
目前的智能手機,硬件上已經(jīng)可以媲美幾年前的PC機了,1G內(nèi)存,512M以上內(nèi)存,3.5以上的屏幕,3G,WIFI等等都成為了新的手機的最低標(biāo)準。而Windows Phone也一改以往WM手機硬件差異大的問題,設(shè)定了最低的硬件標(biāo)準。相對于以前系統(tǒng),性能上,操作上,流暢度上也有了很大提高。但是電池的發(fā)展遠遠跟不上手機的耗電量。大的也就1500MA的電池,最多也就使用1天多,大部分每天都用充電。為了節(jié)約電量,各個平臺的手機都推出了一些省電的措施。
對于Windows Phone來說,剛推出時和iPhone第一版一樣,不支持多任務(wù),一方面是為了給前臺程序提供更多的資源,更流暢的體驗,另一方面也是為了介紹電池的消耗。同事采用了消息推送機制來完成一部分后臺操作,也使用了一種名為墓碑機制來對實現(xiàn)所謂的了“偽多任務(wù)”。 到了最新的“芒果”系統(tǒng),已經(jīng)支持了多任務(wù),但也不同于以往WM說或PC上的多任務(wù),也改進了墓碑機制,加快了程序間的切換。我們這一篇文章就了解下Windows Phone平臺的偽多任務(wù)。
一、偽多任務(wù)
在我們使用PC和WM系統(tǒng)時,多任務(wù)對我們來說是理所應(yīng)當(dāng)?shù)?。一邊聽歌,一邊上QQ,還能上微博發(fā)照片??傻搅薟indows Phone上卻不支持多任務(wù)了,對于一個新的智能系統(tǒng)來說,確實無法讓人接受。但是多任務(wù)存在一個問題就是如果開的任務(wù)過多,系統(tǒng)用起來就會比較卡,在WM系統(tǒng)中這種情況比較常見。為了提供用戶的體驗,目前的一些智能系統(tǒng)都放棄了這種傳統(tǒng)的多任務(wù)方式,而采用了偽多任務(wù),有限制的多任務(wù)和消息推送這些方法。
所謂的偽多人他并不是真正的多任務(wù),只是通過一些處理,讓用戶使用起來感覺不到和多任務(wù)有什么區(qū)別。在Windows Phone中采用了墓碑機制來實現(xiàn)偽多任務(wù)。墓碑這個名字也很恰當(dāng)。當(dāng)我們運行了一個程序后,切換到另一個程序,這時因為是單任務(wù),第一個程序會被終止掉,但是采用墓碑機制,保存了程序當(dāng)前的狀態(tài)和數(shù)據(jù),當(dāng)我們在切換回的時候,雖然是啟動了一個新的實例,但是可以通過墓碑來恢復(fù)到之前的狀態(tài),所以用戶感覺不到有什么區(qū)別。
墓碑機制是系統(tǒng)自動完成的,但是數(shù)據(jù)的保存和恢復(fù)則是我們通過代碼控制的。另外墓碑機制也不是萬能的,因為此時程序是終止的,如果有一些下載,播放,計時的功能就無法通過墓碑機制完成到了。Windows Phone 7則通過推送機制來解決部分問題,而WP7.1 SDK中也提供了一些后臺操作來解決這些問題,這一篇文章我們主要介紹墓碑機制。
二、程序執(zhí)行模型
在深入了解墓碑機制之前我們必須弄清楚程序的執(zhí)行模型。下面這個圖就是Windows Phone 7.1中最新的執(zhí)行模型。此圖來自于MSDN,但是他的圖有個錯誤,修改了下。
對于一個Windows Phone程序來說,他有啟動---運行---休眠---墓碑---退出等5個狀態(tài),狀態(tài)切換之間存在一些事件和方法。下面就具體介紹一個程序從啟動到終止的過程。
1.啟動一個程序
我們啟動一個程序有多種方法,最普遍的就是在第一屏界面點擊程序的Tile圖標(biāo)或者是在第二屏中點擊程序圖標(biāo),另外我們可以通過點擊一個推送的Toast消息或者通過Launchers 和 Choosers來啟動程序,這些后面會介紹。在啟動程序是,會觸發(fā)Application的Launching事件。這個事件我們前面提到過,他是PhoneApplicationService類中注冊的事件,除此之外好包含了三個和程序狀態(tài)相關(guān)的事件。
程序啟動會會創(chuàng)建一個新的實例,為了保證程序能過正常快速的啟動,我們不應(yīng)該在Launching事件中執(zhí)行過多的代碼,比如讀取文件,網(wǎng)絡(luò)連接等等,這樣會印象用戶體驗,我們可用另起一個線程來執(zhí)行這些操作。
2.程序運行
程序啟動以后就進入了運行狀態(tài),程序從啟動到進入第一個頁面的過程我們在前面討論頁面導(dǎo)航的文章中詳細介紹過了。通過從配置文件獲得要加載的第一個XAML頁面,讀取XAML文件并生成新的實例,通過Frame顯示出這個Page。我們知道在進入一個Page時會執(zhí)行OnNavigatedTo方法。當(dāng)這些完成后,程序才真正進入了運行狀態(tài)。而OnNavigatedTo方法對于墓碑機制來說非常重要,我們一般在這個方法中進行一些數(shù)據(jù)恢復(fù)的操作。
3.休眠狀態(tài)
這個狀態(tài)是在WP7.1中新加入的一個狀態(tài),是為了提升多個任務(wù)間的切換速度。當(dāng)我們在程序中點擊Win鍵進入到主界面,或者是在程序中使用了Launchers 和Choosers啟動了另一個程序時就會發(fā)生(不是所有都會發(fā)生)。休眠狀態(tài)時,程序停止運行,但是整個進程還是存在于內(nèi)存中。當(dāng)恢復(fù)這個程序時,就不需要創(chuàng)建一個新的實例。這樣就加快了程序恢復(fù)和切換的速度。而且從休眠狀態(tài)恢復(fù)時我們一不需要去恢復(fù)數(shù)據(jù)。而在WP7.1中,我們可以長按Back按鈕,出現(xiàn)程序列表,然后選擇要前臺執(zhí)行的程序。
在切換到其他程序,進入到休眠狀態(tài)之前,會調(diào)用當(dāng)前頁面的OnNavigatedFrom方法,在這個方法中我們可以保存當(dāng)前頁面的一些數(shù)據(jù)狀態(tài);然后會觸發(fā)Application中的Deactivated事件,在這個事件中,我們可以保存一些當(dāng)前程序的數(shù)據(jù)。我們知道,因為進程資源還保存在內(nèi)存中,所以當(dāng)前臺程序使用時,內(nèi)存不足或者不足以讓程序流暢運行,這時系統(tǒng)就會執(zhí)行一些操作來釋放內(nèi)存,此時程序就可能從休眠狀態(tài)變換為下面介紹的墓碑狀態(tài)。
4.墓碑狀態(tài)
如上面說的,在WP7.1中,只有系統(tǒng)資源不夠前臺程序使用時,后臺程序才可能從休眠狀態(tài)進入到墓碑狀態(tài)。這也是WP7.1相對于WP7.0最大的改進。在WP7.0中,系統(tǒng)沒有休眠狀態(tài),仍和后臺程序都是直接進入到墓碑狀態(tài),而OnNavigatedFrom方法和Deactivated事件都是在進入墓碑轉(zhuǎn)臺前觸發(fā)的。一個程序進入到墓碑狀態(tài)時,他的進程被終止掉,但是程序的回退棧中的信息,以及我們保存的一些信息會保留在內(nèi)存中。
為了保證系統(tǒng)資源和性能,Windows Phone對墓碑狀態(tài)也做了一些限制。目前系統(tǒng)中之允許同時存在5個墓碑程序。當(dāng)超過了這個數(shù)量或者因為前臺程序需要更多的資源,那么系統(tǒng)就會完全終止掉我們的程序,包括閃存內(nèi)存中保存的回退棧和數(shù)據(jù)信息。所以對于我們而言,好需要對這種情況進行處理。
5.程序恢復(fù)
當(dāng)我們使用Back按鈕或者長按Back從任務(wù)列表中返回程序,或者是從Launchers 或Choosers返回時,程序就會恢復(fù)執(zhí)行。但是如圖上看到的,我們程序可能是從休眠狀態(tài),也可能是從墓碑狀態(tài)返回的。如果是從休眠狀態(tài)返回時我們不需要做任何恢復(fù)的操作,而從墓碑程序中返回,我們就需要恢復(fù)程序的狀態(tài)和數(shù)據(jù),以便用戶感覺程序是被重寫創(chuàng)建了。
程序恢復(fù)時會觸發(fā)Application類中的Activated事件,我們可以通過檢查IsApplicationInstancePreserved參數(shù)來判斷程序是從休眠狀態(tài)還是墓碑狀態(tài)返回的,在此方法中我們可以用來恢復(fù)之前在Deactivated事件中保存的數(shù)據(jù)。如果是從休眠狀態(tài)恢復(fù),不會重新執(zhí)行構(gòu)造函數(shù),而從墓碑狀態(tài)恢復(fù)時會重新執(zhí)行構(gòu)造函數(shù)。因為Back Stack在墓碑狀態(tài)中還是保存在內(nèi)存中的,所以這是會返回到我們程序退出時所在的頁面,在進入到頁面之前還是會執(zhí)行OnNavigatedTo方法,對于墓碑狀態(tài)恢復(fù)的程序,我們可以在這個方法中來恢當(dāng)前頁面的狀態(tài)數(shù)據(jù)。
6.程序結(jié)束
在Windows Phone的silverlight框架程序中,我們唯一退出程序的方法就是點擊Back鍵,當(dāng)Back Stack中不存在頁面時程序就會退出,目前系統(tǒng)沒有提供任何Exit方法來以代碼的方式結(jié)束程序。在程序退出時會觸發(fā)Application中最后一個事件,Closing。在這個事件中我們可以釋放一些使用的資源,保存數(shù)據(jù)等等,要注意的是,如果一個程序從墓碑狀態(tài)被結(jié)束,是不會觸發(fā)此事件的。MSDN上說關(guān)閉程序的時間被限制在10s,超過這個時間程序會被終止掉。
#p#
三、數(shù)據(jù)保存和恢復(fù)
至此,我們對于Windows Phone的執(zhí)行模型也有了一定了解,即便WP7.1中引入了休眠方式,但是程序還是可能進入到墓碑模式或則被終止掉。所以無論什么情況,我們都要對數(shù)據(jù)進行保存和恢復(fù)。不過我們可以通過一些系統(tǒng)狀態(tài)來判斷是否需要進行數(shù)據(jù)保存和恢復(fù)。
在進入到墓碑狀態(tài)時,數(shù)據(jù)會被保存到內(nèi)存中,系統(tǒng)為我們提供了數(shù)據(jù)字典來保存我們的數(shù)據(jù)。 對于程序來說,系統(tǒng)的狀態(tài)和數(shù)據(jù)我們可以存放到Application.State這個字典這種,而對于單個頁面來說數(shù)據(jù)可以存放到Page.State的字典中。需要注意的是這些字典中存入的數(shù)據(jù)必須是可序列化的數(shù)據(jù)。另外我們也可以把數(shù)據(jù)保存到隔離存儲區(qū)中。
有了數(shù)據(jù)保存的區(qū)域下面就是數(shù)據(jù)保存的時機,在Application相關(guān)的: Launching、Deactivated、Activated和Closing這4個事件中我們可以對系統(tǒng)的數(shù)據(jù)進行保存和恢復(fù),而在每個頁面進入和離開都會發(fā)生的OnNavigatedTo和OnNavigatedFrom,我們可以用來保存和恢復(fù)頁面的數(shù)據(jù)。
1.單頁面的情況
我們看個列子,Demo5中我們新建了2個頁面。在MianPage中有一個TextBox和三個按鈕。點擊Add時,TextBox的值會增加(默認是1),而點擊Next按鈕會導(dǎo)航到Page1頁面,點擊Run Camera按鈕會調(diào)用相機。代碼運行環(huán)境是WP7.1 SDK Beta2。我們在Application的四個事件和頁面的2個方法中只加入運行的Log信息。
- public MainPage()
- {
- Debug.WriteLine("MainPage Constructor");
- camera = new CameraCaptureTask();
- camera.Completed += new EventHandler<PhotoResult>(camera_Completed);
- num = 1;
- InitializeComponent();
- txtNum.DataContext = num;
- }
- private void Button_Click(object sender, RoutedEventArgs e)
- {
- txtNum.DataContext = ++num;
- Debug.WriteLine("Add Method,TextBox is {0}", num.ToString());
- }
我們點擊Add讓TextBox值增加。然后點擊Win鍵回到住界面,然后點擊Back返回程序,并繼續(xù)點Add,程序Log信息如下:
- Application_Launching
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
- Add Method,TextBox is 4
- MainPage OnNavigatedFrom
- Application_Deactivated
- Application_Activated
- MainPage OnNavigatedTo
- Add Method,TextBox is 5
- Add Method,TextBox is 6
很明顯,因為在WP7.1下,程序進入了休眠狀態(tài),所以我們TextBox不會被清空。讓我們在模擬器上來模仿一下墓碑狀態(tài)的情況,需要進行如下設(shè)置就可以了。
然后我們再次運行程序,結(jié)果如下:
- Application_Launching
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
- Add Method,TextBox is 4
- MainPage OnNavigatedFrom
- Application_Deactivated
- The thread '<No Name>' (0xec600c2) has exited with code 0 (0x0).
- The thread '<No Name>' (0xee100aa) has exited with code 0 (0x0).
- The thread '<No Name>' (0xe6a00ae) has exited with code 0 (0x0).
- 'UI Task' (Managed): Loaded 'C:\Program Files\Microsoft.NET\SDK\CompactFramework\v2.0\WindowsCE\mscorlib.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.RuntimeHost.dll'
- 'UI Task' (Managed): Loaded 'System.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.dll'
- 'UI Task' (Managed): Loaded 'System.Net.dll'
- 'UI Task' (Managed): Loaded 'System.Core.dll'
- 'UI Task' (Managed): Loaded 'System.Xml.dll'
- 'UI Task' (Managed): Loaded '\Applications\Install\F5BE083B-4EC0-4192-84B0-EE83323B8240\Install\Demo5.dll', Symbols loaded.
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.dll'
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.Interop.dll'
- 'UI Task' (Managed): Loaded 'System.Runtime.Serialization.dll'
- Application_Activated
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
很明顯看到,在執(zhí)行完Deactivated事件后程序的線程都結(jié)束了,而但我點擊Back按鈕后,重新加載了程序,并且執(zhí)行了MainPage的構(gòu)造函數(shù),而我們TextBox的值也從1開始了。下面看看如何來保存這些數(shù)據(jù)。對代碼修改如下:
- protected override void OnNavigatedFrom(System.Windows.Navigation.NavigationEventArgs e)
- {
- Debug.WriteLine("MainPage OnNavigatedFrom");
- base.OnNavigatedFrom(e);
- State["num"] = num; //保存變量
- }
- protected override void OnNavigatedTo(System.Windows.Navigation.NavigationEventArgs e)
- {
- Debug.WriteLine("MainPage OnNavigatedTo");
- base.OnNavigatedTo(e);
- num = (int)State["num"]; //恢復(fù)變量
- }
我們在MainPage的兩個方法中保存和恢復(fù)數(shù)據(jù),但是運行卻出錯了。因為OnNavigatedTo中的State["num"]為空。因為我們第一次進入頁面時也會執(zhí)行這個方法,這時還沒有保存過num,所以就出錯了。在WP7.0中我們可以使用 State.ContainsKey或者State.TryGetValue來避免這種錯誤的發(fā)生,任何時候我們從字典中讀取數(shù)據(jù)時都必須檢查數(shù)據(jù)的有效性。而在WP7.1中我們有更好的辦法:
- protected override void OnNavigatedFrom(System.Windows.Navigation.NavigationEventArgs e)
- {
- Debug.WriteLine("MainPage OnNavigatedFrom");
- base.OnNavigatedFrom(e);
- if (e.NavigationMode != System.Windows.Navigation.NavigationMode.Back)
- {
- Debug.WriteLine("Save data");
- State["num"] = num; //保存變量
- }
- }
- protected override void OnNavigatedTo(System.Windows.Navigation.NavigationEventArgs e)
- {
- Debug.WriteLine("MainPage OnNavigatedTo");
- base.OnNavigatedTo(e);
- if (e.NavigationMode != System.Windows.Navigation.NavigationMode.New)
- {
- Debug.WriteLine("Recover data");
- object objNum;
- if (State.TryGetValue("num", out objNum))
- {
- num = (int)objNum;
- }
- }
- txtNum.DataContext = num;
- }
我們在看看結(jié)果:
- Application_Launching
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
- Add Method,TextBox is 4
- MainPage OnNavigatedFrom
- Save data
- Application_Deactivated
- The thread '<No Name>' (0xfa800f2) has exited with code 0 (0x0).
- The thread '<No Name>' (0xf9500e6) has exited with code 0 (0x0).
- The thread '<No Name>' (0xe3700ee) has exited with code 0 (0x0).
- 'UI Task' (Managed): Loaded 'C:\Program Files\Microsoft.NET\SDK\CompactFramework\v2.0\WindowsCE\mscorlib.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.RuntimeHost.dll'
- 'UI Task' (Managed): Loaded 'System.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.dll'
- 'UI Task' (Managed): Loaded 'System.Net.dll'
- 'UI Task' (Managed): Loaded 'System.Core.dll'
- 'UI Task' (Managed): Loaded 'System.Xml.dll'
- 'UI Task' (Managed): Loaded '\Applications\Install\F5BE083B-4EC0-4192-84B0-EE83323B8240\Install\Demo5.dll', Symbols loaded.
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.dll'
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.Interop.dll'
- 'UI Task' (Managed): Loaded 'System.Runtime.Serialization.dll'
- Application_Activated
- MainPage Constructor
- MainPage OnNavigatedTo
- Recover data
- Add Method,TextBox is 5
- Add Method,TextBox is 6
數(shù)據(jù)被正常恢復(fù)了,我們可以看到,OnNavigatedTo方法雖然被執(zhí)行了兩次,但是Recover data只有在從墓碑或休眠狀態(tài)返回時才執(zhí)行,而在新建時是沒有執(zhí)行的。因為在WP7.1中開放了NavigationEventArgs 的NavigationMode屬性。當(dāng)我們新建一個頁面進入時,此屬性為New,而當(dāng)我們從墓碑狀態(tài)或其他頁面返回時則是Back,很明顯我們只有在Back時才需要恢復(fù)數(shù)據(jù)。而在OnNavigatedFrom方法中,當(dāng)我們導(dǎo)航到新的頁面時,此屬性為New,而當(dāng)我們回退到前一個頁面時,此屬性為Back,我們知道Windows Phone中只有Back Stack,所以從一個頁面返回到前一個頁面時,這個頁面就會被回收,所以就沒有必要保存數(shù)據(jù)了。而在WP7中我們每次都要進行保存和恢復(fù)操作。
但是這里還有一個問題,就是當(dāng)我們從休眠狀態(tài)返回時,是不需要進行數(shù)據(jù)恢復(fù)的,但是此方法還是會被執(zhí)行,在WP7.1中同樣提供了方法,可以在休眠狀態(tài)返回時不執(zhí)行這些操作。我們在Application的Activated 事件中可以判斷,我們修App代碼如下:
- public static bool IsTombstoning { get; set; }
- private void Application_Activated(object sender, ActivatedEventArgs e)
- {
- Debug.WriteLine("Application_Activated");
- if (e.IsApplicationInstancePreserved)
- {
- IsTombstoning = false;
- }
- else
- {
- IsTombstoning = true;
- }
- }
WP7.1中新增了IsApplicationInstancePreserved屬性來判斷是休眠狀態(tài)還是墓碑狀態(tài)恢復(fù)的。我們在App中定義了一個靜態(tài)的IsTombstoning屬性。在MainPage的OnNavigatedTo方法中使用就行了。有一點要注意的是不要在App的構(gòu)造函數(shù)中對IsTombstoning 賦值,否則從墓碑狀態(tài)返回后會執(zhí)行構(gòu)造函數(shù),另外也不要在頁面中定義一個變量來指示是否是新的實例,因為你需要保存或恢復(fù)這個變量。取消Debug中的墓碑調(diào)試,我們看看結(jié)果。
- Application_Launching
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
- Add Method,TextBox is 4
- Add Method,TextBox is 5
- Add Method,TextBox is 6
- MainPage OnNavigatedFrom
- Save data
- Application_Deactivated
- Application_Activated
- MainPage OnNavigatedTo
- Add Method,TextBox is 7
2.多頁面的情況
前面已經(jīng)了解了單頁面情況下的數(shù)據(jù)恢復(fù),那么如果我們從主頁面切換到其他頁面,這時進入了休眠或墓碑狀態(tài)又會怎么樣呢?因為這是我們當(dāng)前已經(jīng)不是在MainPage頁面了。還是在Demo5中,我們點擊Add后,點擊Next按鈕,進入到Page1,然后點Win鍵,進入主菜單。在Page1中我們沒有進行數(shù)據(jù)存儲和恢復(fù),我們只關(guān)心MainPage中的數(shù)據(jù)。我們就看看墓碑情況,運行結(jié)果如下:
- Application_Launching
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
- Main Pgee Navigate to Page1
- Page1 Constructor
- MainPage OnNavigatedFrom
- Save data
- Page1 OnNavigatedTo
- Page1 OnNavigatedFrom
- Application_Deactivated
- The thread '<No Name>' (0xf88012a) has exited with code 0 (0x0).
- The thread '<No Name>' (0xe220156) has exited with code 0 (0x0).
- The thread '<No Name>' (0xf1d014a) has exited with code 0 (0x0).
- 'UI Task' (Managed): Loaded 'C:\Program Files\Microsoft.NET\SDK\CompactFramework\v2.0\WindowsCE\mscorlib.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.RuntimeHost.dll'
- 'UI Task' (Managed): Loaded 'System.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.dll'
- 'UI Task' (Managed): Loaded 'System.Net.dll'
- 'UI Task' (Managed): Loaded 'System.Core.dll'
- 'UI Task' (Managed): Loaded 'System.Xml.dll'
- 'UI Task' (Managed): Loaded '\Applications\Install\F5BE083B-4EC0-4192-84B0-EE83323B8240\Install\Demo5.dll', Symbols loaded.
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.dll'
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.Interop.dll'
- 'UI Task' (Managed): Loaded 'System.Runtime.Serialization.dll'
- Application_Activated
- Page1 Constructor
- Page1 OnNavigatedTo
- Bace to MainPage
- MainPage Constructor
- Page1 OnNavigatedFrom
- MainPage OnNavigatedTo
- Recover data
- Add Method,TextBox is 4
事實證明,我們前面寫的代碼依舊有效,沒有進行任何修改就完成了任務(wù)。我們看到在導(dǎo)航到Page1之前就進行了數(shù)據(jù)保存。因為OnNavigatedFrom不僅在進入墓碑狀態(tài)前回執(zhí)行,在離開當(dāng)前頁面時也會執(zhí)行。所以,即便是在Page1中進入了墓碑狀態(tài),當(dāng)我們返回程序,并返回到MainPage頁面是,還是會執(zhí)行OnNavigatedTo方法來恢復(fù)數(shù)據(jù),這時的頁面Mode也是Back。所以我們不需要為多頁面做特殊處理。
我們考慮下,如果從Page1頁面不是GoBack,而是使用Navigate方法重新導(dǎo)航到MainPage呢?這有什么關(guān)系,只不過是新建了一個頁面的實例,我們前一個MainPage還在Back Stack中。那么數(shù)據(jù)會恢復(fù)到這個MainPage中嗎,當(dāng)然不會,我們使用的State屬性是每個頁面實例特有的。那我我如果想在每個新建的MainPage中都能繼續(xù)進行Add操作呢?
對于這種變態(tài)的需求,一般我是懶得做,肯定是你設(shè)計有問題,但是這里還是演示下如何實現(xiàn)。除了使用Page的State,前面我們還提到使用Application的State,這個是所有頁面共享的。我們在Page中增加一個New按鈕,然后修改MainPage的方法:
- protected override void OnNavigatedFrom(System.Windows.Navigation.NavigationEventArgs e)
- {
- Debug.WriteLine("MainPage OnNavigatedFrom");
- base.OnNavigatedFrom(e);
- Debug.WriteLine("Save data");
- PhoneApplicationService.Current.State["num"] = num; //保存到全局
- }
- protected override void OnNavigatedTo(System.Windows.Navigation.NavigationEventArgs e)
- {
- Debug.WriteLine("MainPage OnNavigatedTo");
- base.OnNavigatedTo(e);
- Debug.WriteLine("Recover data");
- object objNum;
- if (PhoneApplicationService.Current.State.TryGetValue("num", out objNum))
- {
- num = (int)objNum;
- txtNum.DataContext = num;
- }
- }
為了一個變量能在不同實例中同時起作用,并公用,我們修改了頁面的方法。這里保存和恢復(fù)數(shù)據(jù)時都不在判斷是否是New或Back,因為任何情況我們都要保存和恢復(fù),以便在所有頁面中使用。另外我們可以使用隔離數(shù)據(jù)區(qū)來存儲和恢復(fù)。結(jié)果就是,無論我們在Page1頁面從墓碑返回后, New一個MainPage,還是Back,或者是先New然后在返回到前一個MainPage,結(jié)果都是在任何頁面都能對這個數(shù)據(jù)繼續(xù)的使用。
當(dāng)然更好的辦法是在App中定義一個全局的變量,這樣我們不需要在Page的頁面中對這些數(shù)據(jù)進行保存和恢復(fù),只需要在Appliation的事件中進行。這樣代碼會很簡潔。
- public static int Number { get; set; }
- private void Application_Activated(object sender, ActivatedEventArgs e)
- {
- Debug.WriteLine("Application_Activated");
- if (e.IsApplicationInstancePreserved)
- {
- IsTombstoning = false;
- }
- else
- {
- IsTombstoning = true;
- Number = (int)PhoneApplicationService.Current.State["number"];
- }
- }
- private void Application_Deactivated(object sender, DeactivatedEventArgs e)
- {
- Debug.WriteLine("Application_Deactivated");
- PhoneApplicationService.Current.State["number"] = Number;
- }
App中代碼修改如上,我們在Activated和Deactivated中進行數(shù)據(jù)恢復(fù)和保存,因為Activated必定是在Deactivated執(zhí)行后執(zhí)行,所以這里沒有判斷是否存在。但還是應(yīng)該寫的嚴謹一些。下面是MainPage中的代碼:
- protected override void OnNavigatedFrom(System.Windows.Navigation.NavigationEventArgs e)
- {
- Debug.WriteLine("MainPage OnNavigatedFrom");
- base.OnNavigatedFrom(e);
- App.Number = num;
- }
- protected override void OnNavigatedTo(System.Windows.Navigation.NavigationEventArgs e)
- {
- Debug.WriteLine("MainPage OnNavigatedTo");
- base.OnNavigatedTo(e);
- num = App.Number;
- txtNum.DataContext = num;
- }
代碼是非常的簡單,如果不是為了掩飾保存和恢復(fù),我們甚至可以直接對App.Number來操作。相比前一種方法,代碼簡潔了很多。結(jié)果這里就不貼了。
3.使用Chooser的情況
在程序中我們可以使用Launcher或者Chooser來運行系統(tǒng)提供的API功能,比如打電話,發(fā)短信,調(diào)用相機等等。具體的使用我們后面會介紹。一般使用這些API都會導(dǎo)致當(dāng)期那程序變?yōu)楹笈_程序。所以也存在數(shù)據(jù)恢復(fù)的問題,但是對于Chooser來說,他會返回數(shù)據(jù),而Lanucher不會返回數(shù)據(jù),那么這個時候就存在一個新得到的數(shù)據(jù)和被保存的數(shù)據(jù)我們應(yīng)該用哪一個的問題。
看下面的列子,我們調(diào)用相機,返回后得到圖片的大小,把大小填入TextBox。
- private CameraCaptureTask camera;
- public MainPage()
- {
- num = 1;
- camera = new CameraCaptureTask();
- camera.Completed += new EventHandler<PhotoResult>(camera_Completed);
- InitializeComponent();
- txtNum.DataContext = num;
- }
我們先定義相機的Chooser,注意這里要定義成類變量,而不能是局部變量,并且設(shè)置完成事件的處理方法。因為如果是局部變量,當(dāng)系統(tǒng)進入到墓碑狀態(tài)在返回時,這個對象就沒有了,就沒辦法執(zhí)行完成事件了,對于Chooser來說也沒有辦法獲得得到返回的數(shù)據(jù)了。
- private void Button_Click_2(object sender, RoutedEventArgs e)
- {
- Debug.WriteLine("Run Camera");
- try
- {
- camera.Show();
- }
- catch (System.InvalidOperationException ex)
- {
- MessageBox.Show(ex.Message);
- }
- }
- void camera_Completed(object sender, PhotoResult e)
- {
- if (e.TaskResult == TaskResult.OK)
- {
- BitmapImage bmp = new BitmapImage();
- bmp.SetSource(e.ChosenPhoto);
- num = bmp.PixelHeight;
- }
- else
- {
- num = 1024; //模擬器沒法拍照
- }
- }
以上我們點擊Run Camera按鈕時,就調(diào)用Show方法調(diào)出,拍完照片后吧照片的高度設(shè)置到TextBox,因為模擬器拍不了,就手動設(shè)置到1024。頁面的數(shù)據(jù)我們采用第一種,單頁面的保存方式。
- Application_Launching
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
- Run Camera
- MainPage OnNavigatedFrom
- Save data
- Application_Deactivated
- Application_Activated
- Set carmera num
- MainPage OnNavigatedTo
- Add Method,TextBox is 1025
從上面結(jié)果可見程序調(diào)用相機時進入休眠狀態(tài) ,返回后先執(zhí)行了完成函數(shù),Set carmera num后數(shù)據(jù)變成了1024。因為是休眠不需要恢復(fù)狀態(tài),所以在此加1后變?yōu)?025。我們在看看從墓碑狀態(tài)返回的結(jié)果。
- Application_Launching
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
- Run Camera
- MainPage OnNavigatedFrom
- Save data
- Application_Deactivated
- The thread '<No Name>' (0xf3802a2) has exited with code 0 (0x0).
- The thread '<No Name>' (0xe0a02da) has exited with code 0 (0x0).
- The thread '<No Name>' (0xea002da) has exited with code 0 (0x0).
- 'UI Task' (Managed): Loaded 'C:\Program Files\Microsoft.NET\SDK\CompactFramework\v2.0\WindowsCE\mscorlib.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.RuntimeHost.dll'
- 'UI Task' (Managed): Loaded 'System.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.dll'
- 'UI Task' (Managed): Loaded 'System.Net.dll'
- 'UI Task' (Managed): Loaded 'System.Core.dll'
- 'UI Task' (Managed): Loaded 'System.Xml.dll'
- 'UI Task' (Managed): Loaded '\Applications\Install\F5BE083B-4EC0-4192-84B0-EE83323B8240\Install\Demo5.dll', Symbols loaded.
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.dll'
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.Interop.dll'
- 'UI Task' (Managed): Loaded 'System.Runtime.Serialization.dll'
- Application_Activated
- MainPage Constructor
- Set carmera num
- MainPage OnNavigatedTo
- Recover data
- Add Method,TextBox is 4
我們發(fā)現(xiàn)從墓碑狀態(tài)返回后,雖然也進行了Set carmera num,但是還進行了Recover data,最新得到數(shù)據(jù)被之前恢復(fù)的數(shù)據(jù)覆蓋了。我們看到返回后先執(zhí)行了頁面的構(gòu)造函數(shù),然后執(zhí)行了相機的完成方法,這里要注意的是,不要在這里操作控件,因為此時Load事件還沒有執(zhí)行,這里操作控件會拋出一個空引用異常。
獲得的數(shù)據(jù)和保存的數(shù)據(jù)間的選擇是比較容易碰到的問題。我們要根據(jù)自己的邏輯來選擇解決的方法,這里我們使用最新的數(shù)據(jù)來代替保存的數(shù)據(jù)。所以修改恢復(fù)數(shù)據(jù)的部分:
- protected override void OnNavigatedTo(System.Windows.Navigation.NavigationEventArgs e)
- {
- Debug.WriteLine("MainPage OnNavigatedTo");
- base.OnNavigatedTo(e);
- //單頁面情況
- if (App.IsTombstoning)
- {
- if (e.NavigationMode != System.Windows.Navigation.NavigationMode.New)
- {
- if (num == 1)
- {
- Debug.WriteLine("Recover data");
- object objNum;
- if (State.TryGetValue("num", out objNum))
- {
- num = (int)objNum;
- }
- }
- }
- }
- txtNum.DataContext = num;
- }
我們知道如果進入到墓碑狀態(tài)返回后,num的數(shù)據(jù)會丟失,而我們在構(gòu)造函數(shù)中把num設(shè)置為1。因為在調(diào)用OnNavigatedTo之前會執(zhí)行方法相機的返回方法camera_Completed。如果num被設(shè)置為1024就不應(yīng)該恢復(fù)數(shù)據(jù)。所以我們可以在OnNavigatedTo中通過num的值來判斷是否恢復(fù),如果是1就需要恢復(fù),如果不是就說明有新的值。
這里需要注意,如果在 camera_Completed中得到的數(shù)據(jù)就是1,那么還是會被恢復(fù),所以一定根據(jù)情況來。這里只是舉例子。另外我們要注意構(gòu)造函數(shù)中num初始化和camera.Completed事件綁定之間的順序,一定要先初始化數(shù)據(jù),最后在注冊完成事件,否則執(zhí)行完camera_Completed方法后,num又被初始化了。
下面看看執(zhí)行結(jié)果:
- <strong>Application_Launching
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
- Run Camera
- MainPage OnNavigatedFrom
- Save data
- Application_Deactivated
- The thread '<No Name>' (0xe83037a) has exited with code 0 (0x0).
- The thread '<No Name>' (0xeb50372) has exited with code 0 (0x0).
- The thread '<No Name>' (0xf53033a) has exited with code 0 (0x0).
- 'UI Task' (Managed): Loaded 'C:\Program Files\Microsoft.NET\SDK\CompactFramework\v2.0\WindowsCE\mscorlib.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.RuntimeHost.dll'
- 'UI Task' (Managed): Loaded 'System.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.dll'
- 'UI Task' (Managed): Loaded 'System.Net.dll'
- 'UI Task' (Managed): Loaded 'System.Core.dll'
- 'UI Task' (Managed): Loaded 'System.Xml.dll'
- 'UI Task' (Managed): Loaded '\Applications\Install\F5BE083B-4EC0-4192-84B0-EE83323B8240\Install\Demo5.dll', Symbols loaded.
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.dll'
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.Interop.dll'
- 'UI Task' (Managed): Loaded 'System.Runtime.Serialization.dll'
- Application_Activated
- MainPage Constructor
- Set carmera num
- MainPage OnNavigatedTo
- Add Method,TextBox is 1025
- </strong>
和之前的區(qū)別就在于這里沒有恢復(fù)數(shù)據(jù),最后使用的是相機得到的數(shù)據(jù)。對于沒有調(diào)用相機,而是點擊Win導(dǎo)致的程序恢復(fù),這時恢復(fù)的數(shù)據(jù),結(jié)果正確:
- Application_Launching
- MainPage Constructor
- MainPage OnNavigatedTo
- Add Method,TextBox is 2
- Add Method,TextBox is 3
- MainPage OnNavigatedFrom
- Save data
- Application_Deactivated
- The thread '<No Name>' (0xebe0382) has exited with code 0 (0x0).
- The thread '<No Name>' (0xeb203be) has exited with code 0 (0x0).
- The thread '<No Name>' (0xec103d6) has exited with code 0 (0x0).
- 'UI Task' (Managed): Loaded 'C:\Program Files\Microsoft.NET\SDK\CompactFramework\v2.0\WindowsCE\mscorlib.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.RuntimeHost.dll'
- 'UI Task' (Managed): Loaded 'System.dll'
- 'UI Task' (Managed): Loaded 'System.Windows.dll'
- 'UI Task' (Managed): Loaded 'System.Net.dll'
- 'UI Task' (Managed): Loaded 'System.Core.dll'
- 'UI Task' (Managed): Loaded 'System.Xml.dll'
- 'UI Task' (Managed): Loaded '\Applications\Install\F5BE083B-4EC0-4192-84B0-EE83323B8240\Install\Demo5.dll', Symbols loaded.
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.dll'
- 'UI Task' (Managed): Loaded 'Microsoft.Phone.Interop.dll'
- 'UI Task' (Managed): Loaded 'System.Runtime.Serialization.dll'
- Application_Activated
- MainPage Constructor
- MainPage OnNavigatedTo
- Recover data
- Add Method,TextBox is 4
#p#
四、禁用墓碑
如果你擁有一臺真機,在升級到到芒果之前,或許總是要忍受恢復(fù)程序時的"Resuming"的提示。當(dāng)然也不是沒有辦法,網(wǎng)上就流傳了修改注冊表來禁用墓碑狀態(tài)的方法。首先你需要有一臺已經(jīng)越獄的機器,然后下載一個RegistryEditor的XPA軟件包。運行這個軟件,把\HKLM\Software\Microsoft\TaskHost\DehydrateOnPause的鍵值從3修改為0。再試試,程序切換的很快了,"Resuming"基本也不出現(xiàn)了。我買手機的時候Mango的Beta版已經(jīng)出了,而且注冊表也已經(jīng)被賣家修改好了。所以在手機上還沒見到"Resuming"就升級到Mango了。在我升級到7712之后,偶爾卻會出現(xiàn)"Resuming",所以決定研究下修改這個注冊表的作用。
搜索了好久,終于找到了一篇相關(guān)的文章:Tombstoning, Dehydration, and Windows Phone 。從文章中我們可以了解到,這個鍵值有4個選項:
1、Don’t dehydrate;
2、Forcefully dehydrate;
3、Gracefully dehydrate;
4、Automatically decide。
表示程序在暫停時是否執(zhí)行Dehydrate,Dehydrate愿意是脫水。在進入墓碑狀態(tài)后,一些系統(tǒng)級別的操作會停止,運行環(huán)境開始回收資源,其中就包括.NET運行時,回收完以后整個程序的運行環(huán)境也被回收了。這樣一個過程就叫做Dehydrate。
系統(tǒng)默認是3,表示命令程序的運行環(huán)境在合適的時候優(yōu)雅的執(zhí)行Dehydrate。但是目前并不清楚是如何決定是否合適。但是對運行環(huán)境來說默認是執(zhí)行Dehydrate,但是也可以選擇不進行Dehydrate。
HRESULT SHSetAutoDehydratingHostEligibility(BOOL fEligible);
通過這個方法可以設(shè)置程序運行環(huán)境是否進行Dehydrate。當(dāng)我們選擇2時,系統(tǒng)總是會執(zhí)行Dehydrate,而選擇1時,系統(tǒng)會給程序發(fā)送一個WM_CLOSE消息來強制,理論上來說我們可以捕獲這個消息,來自己決定是否執(zhí)行Dehydrate。而當(dāng)設(shè)置為0是,Dehydrate被禁用了,也就是我們程序的運行環(huán)境不會被回收,.NET運行時,程序的實例都不會被銷毀。而在我們恢復(fù)程序是,主要是創(chuàng)建新的實例和.NET運行時,導(dǎo)致了出現(xiàn)"Resuming"。
所以我們修改注冊表為0后,系統(tǒng)看起來就系象7.1中的休眠狀態(tài)一樣了。而且程序恢復(fù)也不會創(chuàng)建新的實例,所以并不清楚和WP7.1的休眠有什么區(qū)別。也有可能7.1的休眠也是采用這種方式,默認不執(zhí)行Dehydrate,而在資源不足時執(zhí)行Dehydrate進入墓碑。這樣在恢復(fù)時會有Resuming字樣,而在WP7.0中不執(zhí)行Dehydrate,當(dāng)資源不夠時程序會被結(jié)束,所以不會出現(xiàn)Resuming字樣。這是我個人的猜測。在Mango中用不了注冊表工具,沒法查看。也不知道此鍵值是否還效或者是否有修改。
另外系統(tǒng)規(guī)定只能有5個程序進入墓碑狀態(tài),我在真機上看了下,最多也只能有5個第三方的程序運行,而不管是休眠還是墓碑狀態(tài)。當(dāng)你開第六個程序時,最早使用的程序就會被關(guān)閉,大家可以長按Back按鈕來觀察。
五、總結(jié)
通過上面介紹我們發(fā)現(xiàn)對于我們程序而言,墓碑機制中對數(shù)據(jù)的保存和恢復(fù)是我們需要關(guān)注的地方。我們通過三種情況,介紹了對頁面已經(jīng)程序數(shù)據(jù)的保存和恢復(fù)方法,以及決定是否恢復(fù)數(shù)據(jù)的一般方法。了解了程序的執(zhí)行模型。其中Page中的OnNavigatedTo和OnNavigatedFrom是最重要的方法。另外所有保存的數(shù)據(jù)必須可以被序列化。
下面是微軟關(guān)于執(zhí)行模型的最佳時間方法:http://msdn.microsoft.com/zh-cn/library/ff817009.aspx
其中介紹了一些文章中沒有涉及的部分,比如當(dāng)程序從墓碑模式被終止的時候需要保存數(shù)據(jù)到隔離區(qū)。所以我們應(yīng)該在Deactivated事件中就進行這樣的操作,因為我們無法預(yù)料程序可能進入的狀態(tài)。另外有一點要指出,當(dāng)我們程序被切換至后臺,進入到休眠或墓碑狀態(tài)時,此時我們點擊程序圖標(biāo),重新啟動一個實例時,之前的實例的數(shù)據(jù)都無法恢復(fù)。想解決這個問題只能把數(shù)據(jù)存放到隔離區(qū),然后恢復(fù)。但是會出現(xiàn)歡迎界面,對于有登陸界面的的還會出現(xiàn)登陸界面,我們需要進行更多的處理,才能讓用戶感覺不到是新開的程序。另外其中還建議Application中的事件以及頁面中的兩個方法操作的時間不要超過10s,否則程序會被終止。但是我嘗試了下使用Thread.Sleep(15000),程序并沒有被終止。