Windows Phone開發(43):推送通知第一集之Toast推送
好像有好幾天沒更新了,抱歉抱歉,最近“光榮”地失業,先是忙于尋找新去處,唉,暫時沒有下文。而后又有一些瑣事要辦,不過不要緊,今天咱們繼續。
動畫的內容就告一段落,本系列文章只作簡單引導,不會覆蓋每一個細節,最終能不能學好,就完全看各位自己了。
從本節開始,我們將討論推送通知,這個東西不太好理解,而推送通知的原理和過程,如果你看MSDN的示意圖,相信你會有點暈,若不,我帖出給你看看。
算了,不帖,不知怎么回事,上傳不了圖片。
現補上圖片。
那么,我就說一個故事吧,希望能幫助你理解何為推送通知。
上大學的時候,我很喜歡到圖書館借書,然后,晚上在宿舍里看,一直看到累了就睡覺。有一回,我發現一本好書叫《中國式商道》,結果呢,去圖書館沒找著,但是查一下是有的,我很看這本書,就去問管理員,管理員說可能被別人借了。
這時候我心里想:那就每天來看一下有沒有在書架上,有再借。
管理員似乎猜到了我的心思,他說:“這位同學,你可以留下借書證號和聯系方式,如果你真想看那本書,一旦有人來還書了,我馬上通知你,你不必天天來找。”
我連忙說謝謝。
比如,我開好了應用程序A,用戶B的手機正在使用我的應用程序,但有時候我會發一些通知給用戶B手機,例如,增加新功能或修復某些Bug,或者有公益活邀請用戶參加等。但是,用戶B上的應用程序如何才知道有新消息呢?
按照傳統的做法,在應用程序中做一個定時“炸彈”,每隔一段時間通過網絡訪問一下我的服務器,檢索一下有沒有新消息,然后把結果返回給客戶端應用程序。你想想,這樣做的缺點是什么?
經常訪問網絡,增加網絡流量,也會消耗一定的電量和資源,如果我用GPRS上網,那就倒霉了。
但是,如果我的客戶端從來不需要主動訪問網絡呢,我也不必在應用程序中放置計時器,程序無須訪問網絡,我的新消息不是發送到用戶手機,而是發送到微 軟的云服務器,然后由云服務器把消息推送到用戶手機。這樣就好比前面的例子,我不用天天跑去圖書館找書,只要有那本書,圖書館管理員就把電話找我。你說, 這樣是不是既省心也省力了?
推送通知有三種:Toast通知,磁帖通知和自定義通知。前面兩種都是死的,都是被硬性規定的,你不要問為什么,記住就行了。而第三種即Raw通知,這種通知方式比較靈活,你可以自定義其格式和內容。
今天,我們來了解第一種通知——Toast。
這是什么呢?
本想截個圖的,但不知道啥事,就是上傳不了,沒反應,CSDN的博客經常出問題。那沒辦法了,我用文字描述一下吧,Toast通知就是在應用程序沒有在前臺運行時,如果收到Toast通知,會在屏幕最上方顯示一條提示信息,就和我們收到短信時一樣。
微軟的云服務器會為我們的手機分配一個URL,就在侈的應用程序注冊推送通道后更新的,云服務器就是利用這個URL來找到你的手機并把通知發到手機 上,就像前面例子中,我 留下借書證編號和電話號碼,到時候,管理員可以通過手機號碼來聯系我。實際開發在,你可以通過各種方式把這個URL傳到你的服務器上保存,因為發送推送通 知是需要這個URL的。
一般來說,如果你建有自己的服務器,就應該會有一個固定的IP地址或域名,你不妨通過HTTP方式把用戶手機的URL發送到你的服務器保存。
那么,如何發送推送通知呢?不要被嚇倒,其實很簡單,就是平常我們熟悉的POST方式提交一個HTTP請求罷了,而提交的URL就是從云服務器中得到的URL。而POST的內容就是一個XML文檔。Toast推送通知的格式如下:
- <?xml version="1.0" encoding="utf-8" ?>
- <wp:Notification xmlns:wp="WPNotification">
- <wp:Toast>
- <wp:Text1>文本一</wp:Text1>
- <wp:Text2>文本二</wp:Text2>
- <wp:Param>參數</wp:Param>
- </wp:Toast>
- </wp:Notification>
這是固定的格式,不要問我為什么,它就是死的。“文本一”指的是顯示Toast提示的標題,“本文二”自然就是正文了,文字盡量簡單,最好幾個字搞定。
而“參數”呢?它其它是一個URI,這個URI就是當用戶點擊了Toast消息后啟動應用程序時導航到的頁面,這個與前面我們說到的“次要磁帖”是一樣的。舉幾個例子吧。
/MainPage.xaml
/MainPage.xa/Mml?v=12345
/MainPage.xaml?value1=123&value2=abcd
最后一條其實就是value1=123&value2=abc,別忘了是XML文檔,字符&是要轉義的,記得前面有人提問,在導航 那一節中,在XAML中設置導航頁面/myPage.xaml?t1=aaaa&t2=bbbb,時會報錯,要知道XAML其實就是XML擴展而 來的,特殊字符記住要轉義。
例如,我要發一條Toast通知,標題為“你好”,內容為“想請你吃飯”,參數為“/MainPage.xmal”,那么,我們POST的XML文檔應當為:
- <?xml version="1.0" encoding="utf-8" ?>
- <wp:Notification xmlns:wp="WPNotification">
- <wp:Toast>
- <wp:Text1>你好</wp:Text1>
- <wp:Text2>想請你吃飯</wp:Text2>
- <wp:Param>/MainPage.xaml</wp:Param>
- </wp:Toast>
- </wp:Notification>
知道這一點就好辦了,下面我們來做一個發送Toast消息的服務器端。
1、任你喜歡用哪個版本的VS,新建一個Windows應用程序,很熟悉了吧,就是WinForm。
2、接著是界面,暈了,上傳不了圖片。這樣吧,你隨便扔幾個TextBox上去,分別用來填RUI,第一個值,第二個值,參數,響應消息??偣?個,最后一個用來顯示發送結果,內容較多,建議用多行。再放一個按鈕,觸發它的Click事件,點擊后立即發送。
好,我直接把所有代碼帖上,這東西不好講解,但相信你如果基礎學得好,肯定看得懂。
- using System;
- using System.Collections.Generic;
- using System.ComponentModel;
- using System.Data;
- using System.Drawing;
- using System.Linq;
- using System.Text;
- using System.Windows.Forms;
- using System.Net;
- using System.IO;
- namespace SendToast
- {
- public partial class Form1 : Form
- {
- public Form1()
- {
- InitializeComponent();
- }
- private void btnSend_Click(object sender, EventArgs e)
- {
- HttpWebRequest myRequest = (HttpWebRequest)WebRequest.Create(txtUrl.Text);
- myRequest.ContentType = "text/xml";
- myRequest.Headers.Add("X-WindowsPhone-Target", "toast");
- /*
- * X-NotificationClass 處理間隔
- * 2 - 立即發送
- * 12 - 450秒內發送
- * 22 - 900秒內發送
- */
- myRequest.Headers.Add("X-NotificationClass", "2");
- // 要發送的內容
- string toastMessage = "<?xml version=\"1.0\" encoding=\"utf-8\"?>" +
- "<wp:Notification xmlns:wp=\"WPNotification\">" +
- "<wp:Toast>" +
- "<wp:Text1>" + txtValue1.Text + "</wp:Text1>" +
- "<wp:Text2>" + txtValue2.Text + "</wp:Text2>" +
- "<wp:Param>" + txtParam.Text + "</wp:Param>" +
- "</wp:Toast>" +
- "</wp:Notification>";
- byte[] buffer = Encoding.UTF8.GetBytes(toastMessage);
- myRequest.ContentLength = buffer.Length;
- myRequest.Method = "POST";
- using (Stream stream = myRequest.GetRequestStream())
- {
- stream.Write(buffer, 0, buffer.Length);
- }
- // 接收回應
- HttpWebResponse myResponse = (HttpWebResponse)myRequest.GetResponse();
- string headers= "";
- foreach (var hd in myResponse.Headers.AllKeys)
- {
- headers += hd + " : " + myResponse.Headers[hd] + " | ";
- }
- headers += "\r\n";
- string msg = "";
- using (Stream recStream = myResponse.GetResponseStream())
- {
- StreamReader reader = new StreamReader(recStream, Encoding.UTF8);
- msg = reader.ReadToEnd();
- reader.Close();
- }
- msg += "\r\n\r\n";
- txtResult.AppendText(headers + msg);
- }
- }
- }
接下來,到WP客戶端,同樣隨便你用什么版本的VS,新建一個Silverlight for Windows Phone應用程序,有些人腦子比較敏感,看到Silverlight字樣不知發生什么事。其實,只是了解它的人不多而已,Silverlight其實有 很多優點的,慢慢體會吧,用客觀公正的視角去體會吧。
界面布局就好辦了,我直接上XAML,如果你看不懂,回去復習WPF。
- <Grid x:Name="ContentPanel" Grid.Row="1" Margin="12,0,12,0">
- <TextBlock Name="txtInfo" TextWrapping="Wrap"/>
- </Grid>
后臺代碼也照帖了。
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Net;
- using System.Windows;
- using System.Windows.Controls;
- using System.Windows.Documents;
- using System.Windows.Input;
- using System.Windows.Media;
- using System.Windows.Media.Animation;
- using System.Windows.Shapes;
- using Microsoft.Phone.Controls;
- using Microsoft.Phone.Notification;
- namespace WPApp
- {
- public partial class MainPage : PhoneApplicationPage
- {
- // 構造函數
- public MainPage()
- {
- HttpNotificationChannel myChannel = null;
- // 推送信道的名字,隨便取一個就行了
- string ChannelName = "ToastChannel";
- InitializeComponent();
- // Find靜態方法可以根據名字查找信道
- myChannel = HttpNotificationChannel.Find(ChannelName);
- // 如果找不到,就要創建一個了
- if (myChannel == null)
- {
- myChannel = new HttpNotificationChannel(ChannelName);
- // 注冊事件
- myChannel.ChannelUriUpdated += new EventHandler<NotificationChannelUriEventArgs>(myChannel_ChannelUriUpdated);
- myChannel.ErrorOccurred += new EventHandler<NotificationChannelErrorEventArgs>(myChannel_ErrorOccurred);
- myChannel.ShellToastNotificationReceived += new EventHandler<NotificationEventArgs>(myChannel_ShellToastNotificationReceived);
- // 打開信道
- myChannel.Open();
- // 綁定Toast通知,這樣在程序不在前臺時才會顯示
- // 屏幕上方的通知提示條
- myChannel.BindToShellToast();
- }
- else
- {
- // 如果存在,還要注冊一次事件,因為在程序被扔到后臺后可能會刪除事件綁定
- myChannel.ChannelUriUpdated+=new EventHandler<NotificationChannelUriEventArgs>(myChannel_ChannelUriUpdated);
- myChannel.ErrorOccurred+=new EventHandler<NotificationChannelErrorEventArgs>(myChannel_ErrorOccurred);
- myChannel.ShellToastNotificationReceived+=new EventHandler<NotificationEventArgs>(myChannel_ShellToastNotificationReceived);
- // 在“輸出”窗輸出URL,因為我們只是測試,這樣一來方便一點
- System.Diagnostics.Debug.WriteLine("通道URI為:{0}", myChannel.ChannelUri.ToString());
- }
- }
- void myChannel_ShellToastNotificationReceived(object sender, NotificationEventArgs e)
- {
- string msg = "";
- foreach (string key in e.Collection.Keys)
- {
- msg += key + " : " + e.Collection[key] + "\r\n";
- }
- Dispatcher.BeginInvoke(() =>
- {
- this.txtInfo.Text = msg;
- });
- }
- void myChannel_ErrorOccurred(object sender, NotificationChannelErrorEventArgs e)
- {
- Dispatcher.BeginInvoke(() => MessageBox.Show(e.Message));
- }
- void myChannel_ChannelUriUpdated(object sender, NotificationChannelUriEventArgs e)
- {
- // 當URL發生改變后,還要輸出一次
- // 保證我們得到的是最新版本的URI
- Dispatcher.BeginInvoke(() =>
- {
- System.Diagnostics.Debug.WriteLine("通道URI:{0}", e.ChannelUri.ToString());
- });
- }
- // 這個方法不用我多介紹了
- protected override void OnNavigatedTo(System.Windows.Navigation.NavigationEventArgs e)
- {
- base.OnNavigatedTo(e);
- if (NavigationContext.QueryString.ContainsKey("toastmsg"))
- {
- this.txtInfo.Text = NavigationContext.QueryString["toastmsg"];
- }
- }
- }
- }
好了,那么,如何測試呢,毫無疑問,兩個程序要同時運行,從VS的“輸出”窗口中把RUI復制到發送程序對應的文本框中,填好幾個參數,如標題正文等,然后,你回到WP模擬器,點擊“開始”按鈕,讓應用程序不在最前臺。
再回到服務器端,點擊發送按鈕,等一會兒,你在模擬器中會看到Toast提示條的出現了。
沒辦法上傳圖片,只能這樣了。
下面,總結一下,推送通知其實不難的,其本質就是HTTP通信,而且三種方式有兩種是固定格式的,打開MSDN的示例,照抄就行了,一樣的。
但要理解它不是那么容易,記住要多練,學編程沒什么捷徑,最快的捷徑就是動手干活。你可能會問:你是怎么熟悉這些技術的?
那我告訴你吧,這幾個推送通知的代碼,我已經寫了十幾二十遍了,你說我會不理解嗎?不信你也寫上十遍看看。