GPRS资费受3G的影响逐渐降低,目前5元包月即可以获得30M的流量,而cmwap日渐式微大有被cmnet取代之势(北京GPRS套餐其流量已不区分cmwap和cmnet),而后者可无障碍地和因特网互联互通,这种变化更促进了GPRS的进一步应用。
原以为一旦GPRS连接建立,手机与接入因特网的PC通信和PC之间的通信一样简单,通过简单配置后就可以用socket进行互联,后来深入研究才发现暗礁重重,要想完美实现手机和PC的互联互通还真得费一番功夫。
试验平台如下:
1、 智能手机Windows Mobile 6.0;
2、 一台通过ADSL上网的PC(动态公网IP,每次拨号连接后的IP都很随机);
3、 一个个人网站空间;
实现思路如下:
1、 开发一个运行在PC上的网络服务程序(服务端),功能包括:获取PC拨号后的动态公网IP,把该IP和端口信息上传到有固定域名的个人网站空间(当然这个功能也可以用花生壳之类的工具完成域名到动态IP的绑定);提供特定的功能的网络服务(如让家中的电脑执行如下载指定的文件,打开摄像头等任务,甚而如果PC连接了家庭控制系统,可以遥控家里的空调、热水器等家电进行工作)。
2、 开发运行在Windows Mobile 5.0/6.0上的客户端控制程序,功能包括:可从指定URL获得PC服务程序上传的IP和端口信息;通过获得的IP、端口和PC上的服务程序进行连接并通信,从而得以远程控制居家的PC。
网络拓扑图:
说明:① PC上传公网IP和端口号到Web服务器;
② 手机从Web服务器获取PC的IP和端口号;
③ 手机和PC直接通信互联;
实际效果图:
说明:① 选择GPRS接入点(要选择Internet设置),并接入;
② 探测居家PC的IP和端口(从Web Server获取);
③ 连接居家PC,并进行通信测试;
说明:PC服务程序
相关代码部分:
1、 GPRS连接
相关代码已有很多网上文章进行了介绍,本部分的代码主要来自于网络,并进行了些微调整,详情可参见:http://www.cnblogs.com/mugua/archive/2009/04/11/1433509.html
本打算让程序也可以通过cmwap进行连接,调试时程序也可以和代理服务器10.0.0.127:80进行连接,但是相关的Http请求总是返回出错,查看了相关文章,估计移动对10.0.0.127代理还是做了一定的限制的,幸好cmnet目前已可取代cmwap,所以我们姑且先用cmnet方式吧,不过如果在cmwap上网友有更好的解决方案,也希望能share一下。
2、 GPRS文件下载(下载IP和端口信息)
奇怪的是下列代码当手机通过Microsoft ActiveSync连接时,可以正确获取,但是建立GPRS连接后则代码执行失败。
- public static string DownLoadFile(string fileURL)
- {
- StreamReader rdr = null;
- FileStream wrtr = null;
- string localFileName = string.Empty;
- try
- {
- HttpWebRequest req = (HttpWebRequest)WebRequest.Create(fileURL);
- req.Method = "GET";
- HttpWebResponse resp = (HttpWebResponse)req.GetResponse();
- long len = resp.ContentLength;
- Stream resprespStream = resp.GetResponseStream();
- localFileName = @"\Program Files\" + fileURL.Substring(fileURL.LastIndexOf("/") + 1);
- wrtr = new FileStream(localFileName, FileMode.Create);
- byte[] inData = new byte[4096];
- int bytesRead = respStream.Read(inData, 0, inData.Length);
- while (bytesRead > 0)
- {
- wrtr.Write(inData, 0, bytesRead);
- bytesRead = respStream.Read(inData, 0, inData.Length);
- }
- //判断下在是否成功
- System.IO.FileInfo fi = new FileInfo(localFileName);
- if (fi.Length != len) localFileName = string.Empty;
- fi = null;
- }
- catch {}
- finally
- {
- if (rdr != null) rdr.Close();
- if (wrtr != null) wrtr.Close();
- }
- return localFileName;
- }
最后没有办法,我只好自己实现了一个基于HTTP协议下载的程序,相关代码如下:
public string HttpDownload(string url)
{
try
{
url = url.ToLower().Replace("http://", "");
int offset = url.IndexOf("/");
string Host=url.Substring(0,offset);
string file = url.Substring(offset);
IPHostEntry IpHost = Dns.GetHostEntry(Host);
Socket Sock = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
IPEndPoint point = new IPEndPoint(IpHost.AddressList[0], 80);
Sock.Connect(point);
if (Sock.Connected)
{
string strGet = "GET " + file + " HTTP/1.0\r\n" +
"Host:" + Host + "\r\n" +
"Accept:*/*\r\n" +
"User-Agent:GeneralDownloadApplication\r\n" +
"Connection:Keep-Alive\r\n\r\n";
Byte[] cmd = Encoding.Default.GetBytes(strGet.ToCharArray());
Sock.Send(cmd, cmd.Length, SocketFlags.None);
string strInfo = "";
for (int i = 0; i < 1000; i++) //10s
{
if (Sock.Available > 0)
{
Byte[] bytes = new Byte[1024];
int intSize = Sock.Receive(bytes, 1024, 0);
strInfo += Encoding.Default.GetString(bytes, 0, intSize);
if (strInfo.IndexOf("HTTP/1.1 200 OK") == 0)
{
intSize = strInfo.IndexOf("\r\n\r\n");
if (intSize > 0)
{
return strInfo.Substring(intSize + 4);
}
}
}
Thread.Sleep(10);
}
}
}
catch { }
return "";
}
3、 GPRS通信程序,该部分和正常的Socket通信一般无二,所以相关代码略。
4、 服务端信息上传代码。
该部分代码实现比较简单,直接用WebClient类的UploadString可以上传到指定FTP服务器,代码如下:
- public static bool HttpUpLoad(string HostName,string ip,int port)
- {
- WebClient client = new WebClient();
- try
- { client.UploadString("ftp://yefan:[email protected]/" + HostName + ".txt", ip + "|" + port.ToString() + "|" + DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss"));
- return true;
- }
- catch(Exception ex)
- {
- YFBase.ShowInfo(ex.Message.ToString(), "HttpUpLoad", YFBase.InfoTypes.LVS_error);
- }
- return false;
- }
5、 服务端通信代码,和客户端一样,普通的Socket通信,代码略。
以上仅仅是一个简单的demo和粗略的想法,希望能起到抛砖引玉的作用,让更多的网友参与其中,做出更炫更实用的GPRS程序。(作者:叶帆 2009年9月6日)