上一节完成后,程序已经能够正常运行了。基本功能都完成了。是不是有种很不出的感觉呢。大家都应该听说过会这个墓碑机制了吧。因为window phone 是单任务系统,为后台程序留了5个坑。程序一旦进入后台,就休眠了。被埋在坑里了,就不再运行了。只有你按返回键才能从进入休眠的后台程序。不过,我觉得还有两个问题,第一个就是如果程序在后台,你从启动器启动程序,就是不是按返回键进入程序,那么程序就重新运行了。原来后台的数据就没了,这个是一个用户体验很不好的地方,比如我,就是不喜欢按返回键进入程序的,喜欢从启动器进入后台程序。第二个,就是只有5个坑,一旦你的程序进入后台之后,又有5个程序进入后台,那么你程序埋在坑里的尸体就被抛尸荒野了。再也不能从后台进入了,只能从启动器重新打开。。

那么,怎么解决呢,大家都是喜欢多任务运行的。我们就来模拟一把吧。

模拟之前,先要了解程序的生命周期,这些大家还是百度吧。现在只是简单说下涉及到声明周期要使用到的函数。在App.xaml.cs里面定义了那么4个函数,Launching,Activate,Deactivate,Closing。Launching在程序启动的时候执行。Deactivate在用户点击开始键或者其他导致程序进入后台的操作时候执行,Activate在用户按返回键进入程序的时候执行。closing不用多说,看名字就知道。在程序关闭的时候执行。

在我们这个程序,也就在设置页面的时候如果中断运行进入后台的话才需要恢复,所以相对简单些。我的方案是:定义了两个变量,一个是保存当前所在页面的CurrentPage(保存到PhoneApplicationService),一个是保存是否中断到后台的Recovery。.在Deactivate函数里面保存需要的相关值和上面两个变量到Isolatedstorage。并且在Launching的时候把原来Deactivate保存的值取出来。然后再PageLoad 的时候判断是否是中断恢复。是的话就导航到相关页面,然后恢复界面值。恢复完成,Recovery设置为false,并且保存到Isolatedstorage。


在写代码之前,还要了解的是每个Page里面都有两个事件,OnNavigatedFrom和OnNavigatedTo,前者就是离开页面的时候执行,后者是进入页面的时候执行。重载这两个事件就能用了。

下面先修改App.xaml.cs的代码:(代码中可能没添加命名空间,自己加吧,很简单)


IsolatedStorageSettings iss = IsolatedStorageSettings.ApplicationSettings;
        private void Application_Launching(object sender, LaunchingEventArgs e)
        {

            //如果数据库不存在则创建一个数据库,并且存入相关数据

            CreatDB();

            #region 把Deactivate保存的相关数据取出来
            string temp;
            if (iss.TryGetValue("LastPage", out temp))
            {
                PhoneApplicationService.Current.State["CurrentPage"] = temp;
            }
            int index;
            if (iss.TryGetValue("cityIndex", out index))
            {
                PhoneApplicationService.Current.State["cityIndex"] = index;
            }
            if (iss.TryGetValue("provIndex", out index))
            {
                PhoneApplicationService.Current.State["provIndex"] = index;
            } 
            #endregion
        }


        private void Application_Deactivated(object sender, DeactivatedEventArgs e)
        {
            #region 程序即将进入后台,保存中断情况的相关数据
            iss["Recovery"] = true;
            if (PhoneApplicationService.Current.State.ContainsKey("cityIndex"))
            {
                iss["cityIndex"] = PhoneApplicationService.Current.State["cityIndex"];
            }
            if (PhoneApplicationService.Current.State.ContainsKey("provIndex"))
            {
                iss["provIndex"] = PhoneApplicationService.Current.State["provIndex"];
            }
            if (PhoneApplicationService.Current.State.ContainsKey("CurrentPage"))
            {
                iss["LastPage"] = PhoneApplicationService.Current.State["CurrentPage"];
            } 
            #endregion

        }

        private void Application_Closing(object sender, ClosingEventArgs e)
        {
            //退出,
            iss["Recovery"] = false;
        }



下面修改MainPage的代码:(PageLoad修改如下)


private void PhoneApplicationPage_Loaded(object sender, RoutedEventArgs e)
        {
            #region 判断是否是中断进入
            IsolatedStorageSettings iss = IsolatedStorageSettings.ApplicationSettings;
            bool recov;
            if (iss.TryGetValue("Recovery", out recov))
            {
                if (recov)
                {
                    if (PhoneApplicationService.Current.State.ContainsKey("CurrentPage"))
                    {
                        string temp = Convert.ToString(PhoneApplicationService.Current.State["CurrentPage"]);
                        if (!temp.Equals("MainPage.xaml") && temp.Length > 0)
                        {
                            NavigationService.Navigate(new Uri("/WeatherForecast;component/" + temp, UriKind.Relative));
                        }
                        else
                        {
                            iss["Recovery"] = false;
                        }
                    }
                }
                else
                    PhoneApplicationService.Current.State["CurrentPage"] = "MainPage.xaml";
            }
            else
                PhoneApplicationService.Current.State["CurrentPage"] = "MainPage.xaml";
            #endregion
            UpdateWeather();
        }




我们主要做的是SetPage的中断进入后台的处理,所以,要重载OnNavigatedFrom事件来保存两个Listpicker的SelectedIndex。而且,要在PageLoad的时候判断是否是中断返回,如果是那么就恢复。不是就常规绑定。下面是修改或者增加的代码:


private void PhoneApplicationPage_Loaded(object sender, RoutedEventArgs e)
        {
            #region 判定是否是中断退出后进入,是就恢复数据
            IsolatedStorageSettings iss = IsolatedStorageSettings.ApplicationSettings;
            bool recov;
            if (iss.TryGetValue("Recovery", out recov))
            {
                if (recov)
                {
                    if (PhoneApplicationService.Current.State.ContainsKey("CurrentPage"))
                    {
                        string temp = Convert.ToString(PhoneApplicationService.Current.State["CurrentPage"]);
                        if (temp.Equals("SetPage.xaml"))
                        {
                            int provIndex = 0;
                            int cityIndex = 0;
                            if (PhoneApplicationService.Current.State.ContainsKey("provIndex"))
                            {
                                provIndex = Convert.ToInt32(PhoneApplicationService.Current.State["provIndex"]);
                            }
                            if (PhoneApplicationService.Current.State.ContainsKey("cityIndex"))
                            {
                                cityIndex = Convert.ToInt32(PhoneApplicationService.Current.State["cityIndex"]);
                            }
                            RecovListPickerItems(provIndex, cityIndex);//恢复数据
                            iss["Recovery"] = false;//设置为已经恢复

                        }
                        else
                            ProvLpDataBind();
                    }
                    else
                        ProvLpDataBind();
                }
                else
                    ProvLpDataBind();
            }
            else
                ProvLpDataBind();
            #endregion
            PhoneApplicationService.Current.State["CurrentPage"] = "SetPage.xaml";
        }

        /// <summary>
        /// 恢复数据的时候绑定city的ItemSource,并且绑定两个ListPicker的当前选项
        /// </summary>
        /// <param name="provIndex"></param>
        /// <param name="cityIndex"></param>
        void RecovListPickerItems(int provIndex, int cityIndex)
        {
            provincelp.ItemsSource = prov;
            if (provIndex < 0 || provIndex >= prov.Length)
            {
                provIndex = 0;
            }
            provincelp.SelectedIndex = provIndex;
            List<String> l = cityDataBind(prov[provIndex]);
            if (cityIndex < 0 || cityIndex >= l.Count())
            {
                cityIndex = 0;
            }
            citylp.SelectedIndex = cityIndex;

        }

        /// <summary>
        /// 离开页面保存当前两个ListPicker的selectedIndex
        /// </summary>
        /// <param name="e"></param>
        protected override void OnNavigatedFrom(System.Windows.Navigation.NavigationEventArgs e)
        {
            base.OnNavigatedFrom(e);
            PhoneApplicationService.Current.State["provIndex"] = provincelp.SelectedIndex;
            PhoneApplicationService.Current.State["cityIndex"] = citylp.SelectedIndex;
        }



这样,多任务模拟就完成了。无论是从返回键返回程序还是从启动器启动返回程序。都行。不过,如果要做的页面太多,控件太多还真是很麻烦的。

PS:这里的断点调试技巧:由于要测试从启动器恢复程序,但是编译器调试的时候,一旦点击开始按钮进入后台,然后从启动器进入程序,编译器调试就退出了。因为原来的程序已经终结了。其实我们可以这样。点击开始按钮进入后台后,回到编译器点击终止调试,然后再次启动启动。那么这样的话就可以到需要调试的断点了。毕竟这里的判断还是很多,容易出错。其实这个是利用的是Isolatedstorage的特点,模拟器不关闭这个Isolatedstorage的数据是不会丢失的。