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。

网络拓扑图:

手机上如何打开emjsx_网络


说明:①PC上传公网IP和端口号到Web服务器;

② 手机从Web服务器获取PC的IP和端口号;

③ 手机和PC直接通信互联;

实际效果图:

手机上如何打开emjsx_IP_02

说明:① 选择GPRS接入点(要选择Internet设置),并接入;

② 探测居家PC的IP和端口(从Web Server获取);

③ 连接居家PC,并进行通信测试;

手机上如何打开emjsx_服务程序_03

说明:PC服务程序

相关代码部分:

1、 GPRS连接

相关代码已有很多网上文章进行了介绍,本部分的代码主要来自于网络,并进行了些微调整,详情可参见:

本打算让程序也可以通过cmwap进行连接,调试时程序也可以和代理服务器10.0.0.127:80进行连接,但是相关的Http请求总是返回出错,查看了相关文章,估计移动对10.0.0.127代理还是做了一定的限制的,幸好cmnet目前已可取代cmwap,所以我们姑且先用cmnet方式吧,不过如果在cmwap上网友有更好的解决方案,也希望能share一下。

2、 GPRS文件下载(下载IP和端口信息)

奇怪的是下列代码当手机通过Microsoft ActiveSync连接时,可以正确获取,但是建立GPRS连接后则代码执行失败。

publicstatic string DownLoadFile(string
 {
StreamReader rdr = null;
FileStream wrtr = null;
string localFileName = string.Empty;
try
 {
HttpWebRequest req = (HttpWebRequest)WebRequest.Create(fileURL);
"GET";
HttpWebResponse resp = (HttpWebResponse)req.GetResponse();
long
Stream
@"/Program Files/" + fileURL.Substring(fileURL.LastIndexOf("/") + 1);
new FileStream(localFileName, FileMode.Create);
byte[] inData = new byte[4096];
int
while
 {
 wrtr.Write(inData, 0, bytesRead);
 bytesRead = respStream.Read(inData, 0, inData.Length);
 }
//判断下在是否成功
FileInfo fi = new FileInfo(localFileName);
if (fi.Length != len) localFileName = string.Empty;
null;

 }
catch {}
finally
 {
if (rdr != null) rdr.Close();
if (wrtr != null) wrtr.Close();
 }
return
}

最后没有办法,我只好自己实现了一个基于HTTP协议下载的程序,相关代码如下:

publicstring HttpDownload(string
 {
try
 {
"http://", "");
int offset = url.IndexOf("/");
string
string

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
 {
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());
SocketFlags.None);
string strInfo = "";
for (int i = 0; i < 1000; i++) //10s
 {
if
 {
Byte[] bytes = new Byte[1024];
int
Encoding.Default.GetString(bytes, 0, intSize);
if (strInfo.IndexOf("HTTP/1.1 200 OK") == 0)
 {
"/r/n/r/n");
if
 {
return
 }
 }
 }
Thread.Sleep(10);
 }
 }
 }
catch
return "";
 }

3、 GPRS通信程序,该部分和正常的Socket通信一般无二,所以相关代码略。

4、 服务端信息上传代码。

该部分代码实现比较简单,直接用WebClient类的UploadString可以上传到指定FTP服务器,代码如下:

publicstatic bool HttpUpLoad(string HostName,string ip,int
 {
WebClient client = new WebClient();
try
"" + HostName + ".txt", ip + "|" + port.ToString() + "|" + DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss"));
return true;
 }
catch(Exception
 {
YFBase.ShowInfo(ex.Message.ToString(), "HttpUpLoad", YFBase.InfoTypes.LVS_error);
 }
return false;
 }

5、 服务端通信代码,和客户端一样,普通的Socket通信,代码略。

以上仅仅是一个简单的demo和粗略的想法,希望能起到抛砖引玉的作用,让更多的网友参与其中,做出更炫更实用的GPRS程序。(作者:叶帆 2009年9月6日)