GCM(谷歌云推送)客户端服务器端开发全指南(服务器篇)

来源:互联网 发布:mysql select 两个表 编辑:程序博客网 时间:2024/04/28 04:12

由于谷歌云推送GCM升级成为FCM,所以此博客能容仅供参考 ————2016.12.2更新

今天我们按照之前所说的步骤介绍GCM云推送服务端的开发,因为服务端的开发比客户端的开发较简单,遵从由易到难,一步一步攻破的原则,所以我先于客户端讲服务端的开发,话不多说,让我们开始吧!

首先我们依旧来到首页
这里写图片描述
这次我们点击指南,进入到GCM开发Overview,这里概括了GCM客户端服务器端开发流程。

根据以下的流程图我们不难看出服务端和GCM的通信方式有两种
1.Http协议
2.Xmpp协议
这里写图片描述

Xmpp协议常用于双向通信,我们这里暂时不需要,因此果断选择Http协议来开发。

英语比较好的朋友可以看一下key的概念,这里大概介绍了google是怎么样通过key鉴别发送者和接收者的。
不想了解的朋友只需知道我们要用到
1.Sender ID
2.API Key
3.Registration Token
就行了,然后点击“HTTP Connection Server”开始我们的编程
这里写图片描述

既然选择了http协议通信,那你必须了解http协议相关知识,这里不再阐述,有兴趣的朋友可以来这里了解下推荐入手《HTTP权威指南》。

我们接着来看一下google发布的http service开发指南
这里写图片描述

根据上诉文章我们需要:
-设置http的链接为:https://gcm-http.googleapis.com/gcm/send
-设置http头部参数:Authorization: key=YOUR_API_KEY,
-设置Content-Type:如果发送的消息是json格式,就使用Content-Type: application/json for JSON; 如果是纯文本(比如说一条String字符串)就使用Content-Type:application/x-www-form-urlencoded;charset=UTF-8

其实他这里可能漏了两个参数,一个是http方法 Method 和 Sender ID,Method 是使用http一般都要设置的参数,我们这里使用“post”方法,Sender ID就真的是一个坑了,一开始楼主并不知道有这个参数,怎么发消息都发布出去,问了下有经验的大神才知道Sender ID也要给上才行的。

然后这里说一下,服务端的开发我选择使用.net,不得不说,.net对于一些大型项目的支持真的很好,就是VS比较重量级,不是处处可以布置,也许你会喜欢使用java来写,但是个人感觉面向对象编程语言的逻辑是相同的,楼主一开始学习的是java,然后因为需求兼顾了.net的开发,再后来使用的objective-c,使用起来逻辑并不会出现冲突的情况,顶多就是API和语法个别地方有些不一样,使用起来不是很熟手而已。所以看了我用.net写的服务端程序,你一定很快就能用java copy出来的。

接着就是激动人心的时刻——————上代码!

//建立一个http链接并设置http头部参数 WebRequest gcmRequest; gcmRequest = WebRequest.Create("https://gcm-http.googleapis.com/gcm/send"); gcmRequest.Method = "post"; gcmRequest.ContentType = "application/json"; gcmRequest.Headers.Add(string.Format("Authorization: key={0}", API_KEY)); gcmRequest.Headers.Add(string.Format("Sender: id={0}", SENDER_ID));

好了,我们接着往下看
这里写图片描述
这里介绍的是发送消息的格式,分别是json和纯文本的格式,我们来看看json的

{ "collapse_key": "score_update",  "time_to_live": 108,  "delay_while_idle": true,  "data": {    "score": "4x8",    "time": "15:16.2342"  },  "to" : "bk3RNwTe3H0:CI2k_HHwgIpoDKCIZvvDMExUdFQ3P1..."}

参数按照他所说的那样设,要发送的消息写在data内,然后又是一个坑爹的地方“to”,他这个to不知是不是新版本才用到的,我使用的时候他总返回400错误给我,google了一下,把这里的“to”改成“registration_ids”就可以了,修改后的json

{ "collapse_key": "score_update",  "time_to_live": 108,  "delay_while_idle": true,  "data": {    "message": "Hi lady",    "time": "15:16.2342"  },  "registration_ids" : "bk3RNwTe3H0:CI2k_HHwgIpoDKCIZvvDMExUdFQ3P1..."}

好,有格式后我们就可以开始写Http的body部分了

 //把封装好的json打包成byte并以流的形式发送出去 var postData = b.ToString(); System.Diagnostics.Debug.WriteLine(postData); Byte[] byteArray = Encoding.UTF8.GetBytes(postData); gcmRequest.ContentLength = byteArray.Length; Stream dataStream = gcmRequest.GetRequestStream(); dataStream.Write(byteArray, 0, byteArray.Length); dataStream.Close();

写好body后就可以发送出去了

 //建立一个接收链接接收返回的数据 WebResponse tResponse = gcmRequest.GetResponse(); dataStream = tResponse.GetResponseStream(); StreamReader tReader = new StreamReader(dataStream); String sResponseFromServer = tReader.ReadToEnd();

发送出去我怎么知道他是否成功能?让我们继续看下去
这里写图片描述
这里写图片描述
根据说明,返回的是一组json数据,具体数据代表什么呢?我们直接去看他们返回的代码说明,就是Downstream message error response codes

来到新的页面后直接去看返回代码表
这里写图片描述
这里写图片描述

这里我简单说下几个常见的错误:
1.Missing Registration Token 丢失注册ID
-这个错误一般是你的注册ID格式错误,或者在json上没加双引号,逗号之类。
2.Invalid Registration Token 无效的注册ID
-这个错误一般是你ID有误,少字母,多一横之类的。
3.Unregistered Device 没有注册的设备
-这个错误是说你要发送的android手机没有在google service注册过
4.Authentication Error 认证错误
-这里一般会出现在你的json头参数Authentication 设置错误的时候出现。
5.Mismatched Sender 不匹配的Sender ID
-Sender ID有误
6.Invalid JSON JSON格式有误

好了,以上几个问题是我常遇到的,给大家做个参考。

服务器端功能的代码就这样写完了,是不是很简单?
你可以把代码写到webservice上去,这样就能随时调用了。为了方便,我直接做个简单的页面来实现这个功能,在源码上有,截个实现的图片先

这里写图片描述
这里返回了Invalid Registration,就是说“基友A,舍友B,多个ID英文逗号隔开”是无效的ID,这也是理所当然的,让我们来个成功的

这里写图片描述

成功的话他会返回成功ID,就不会有上面的错误代码出现。

以下是功能所有代码:

    public partial class WebForm1 : System.Web.UI.Page    {        private string API_KEY = "你服务器的api——key";        private string SENDER_ID = "你的senderID";        protected void Page_Load(object sender, EventArgs e)        {        }        protected void Button1_Click(object sender, EventArgs e)        {            var message = TextBox1.Text;            var regID = TextBox2.Text;            String[] regIDAll = new String[] { };            regIDAll = regID.Split(',');            //对ID进行双引号的添加            if (regIDAll.Length > 1)            {                regID = "";                for (int i = 0; i < regIDAll.Length; i++)                {                    if (i == 0)                    {                        regIDAll[i] = "\"" + regIDAll[i] + "\"";                    }                    else                    {                        regIDAll[i] = ",\"" + regIDAll[i] + "\"";                    }                    regID = regID + regIDAll[i];                }            }            else            {                regID = "\"" + regID + "\"";            }            //建立一个http链接并设置http头部参数            WebRequest gcmRequest;            gcmRequest = WebRequest.Create("https://gcm-http.googleapis.com/gcm/send");            gcmRequest.Method = "post";            gcmRequest.ContentType = "application/json";            gcmRequest.Headers.Add(string.Format("Authorization: key={0}", API_KEY));            gcmRequest.Headers.Add(string.Format("Sender: id={0}", SENDER_ID));            //封装发送消息参数成json            StringBuilder b = new StringBuilder();            b.Append("{");            b.Append("\"collapse_key\":\"score_update\",");            b.Append("\"time_to_live\": 108,");            b.Append("\"delay_while_idle\": true,");            b.Append("\"data\": {");            b.Append("\"message\": \"" + message + "\",");            b.Append("\"time\": \"" + System.DateTime.Now.ToString() + "\",");            b.Append(" }");            b.Append("\"registration_ids\": [");            b.Append(regID);            b.Append("]");            b.Append("}");            //把封装好的json打包成byte并以流的形式发送出去            var postData = b.ToString();            System.Diagnostics.Debug.WriteLine(postData);            Byte[] byteArray = Encoding.UTF8.GetBytes(postData);            gcmRequest.ContentLength = byteArray.Length;            Stream dataStream = gcmRequest.GetRequestStream();            dataStream.Write(byteArray, 0, byteArray.Length);            dataStream.Close();            //建立一个接收链接接收返回的数据            WebResponse tResponse = gcmRequest.GetResponse();            dataStream = tResponse.GetResponseStream();            StreamReader tReader = new StreamReader(dataStream);            String sResponseFromServer = tReader.ReadToEnd();            //直接在label显示返回的结果            Label1.Text = "result: " + sResponseFromServer;            tReader.Close();            dataStream.Close();            tResponse.Close();        }    }}

以下是我页面的代码:

<!--注意!!这是APS.NET的代码,要在html文件上运行要稍加修改!!--><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"><html xmlns="http://www.w3.org/1999/xhtml"><head runat="server">    <title></title>    <link href="css/AndroidPush.css" rel="Stylesheet" type="text/css" /></head><body><form id="Form1" runat=server>    <div class="t-box">        <div class="box_1">            <div class="t-con">                <img src="logo.png" alt="" />                <div class="abs-box">                    <div class="con-box">                        <div class="con">                            <asp:TextBox ID="TextBox1" runat="server" TextMode="MultiLine" Wrap=true Text="这里输入发送的消息"></asp:TextBox>                            <asp:TextBox ID="TextBox2" runat="server" TextMode="MultiLine" Wrap=true Text="这里输入设备在GCM注册的ID,多个注册ID逗号分开"></asp:TextBox>                            <div class="result">                                <asp:Label ID="Label1" runat="server" Text="result:"></asp:Label>                            </div>                            <div class="operation"><asp:Button ID="Button1" runat="server" Text="Send"                                     onclick="Button1_Click" /></div>                        </div>                    </div>                </div>            </div>        </div>    </div>    </form></body></html>

最后是万众期待的源码!!!!!

源码在此,请叫我雷锋

1 0
原创粉丝点击