让智能手机和居家电脑互联互通(WM6 GPRS)

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连接后则代码执行失败。

 

  
  
  
  
  1. public static string DownLoadFile(string fileURL)  
  2.  
  3.         {  
  4.  
  5.             StreamReader rdr = null;  
  6.  
  7.             FileStream wrtr = null;  
  8.  
  9.             string localFileName = string.Empty;  
  10.  
  11.             try  
  12.  
  13.             {  
  14.  
  15.                 HttpWebRequest req = (HttpWebRequest)WebRequest.Create(fileURL);  
  16.  
  17.                 req.Method = "GET";  
  18.  
  19.                 HttpWebResponse resp = (HttpWebResponse)req.GetResponse();  
  20.  
  21.                 long len = resp.ContentLength;  
  22.  
  23.                 Stream resprespStream = resp.GetResponseStream();  
  24.  
  25.                 localFileName = @"\Program Files\" + fileURL.Substring(fileURL.LastIndexOf("/") + 1);  
  26.  
  27.                 wrtr = new FileStream(localFileName, FileMode.Create);  
  28.  
  29.                 byte[] inData = new byte[4096];  
  30.  
  31.                 int bytesRead = respStream.Read(inData, 0, inData.Length);  
  32.  
  33.                 while (bytesRead > 0)  
  34.  
  35.                 {  
  36.  
  37.                     wrtr.Write(inData, 0, bytesRead);  
  38.  
  39.                     bytesRead = respStream.Read(inData, 0, inData.Length);  
  40.  
  41.                 }  
  42.  
  43.                 //判断下在是否成功  
  44.  
  45.                 System.IO.FileInfo fi = new FileInfo(localFileName);  
  46.  
  47.                 if (fi.Length != len) localFileName = string.Empty;  
  48.  
  49.                 fi = null;  
  50.  
  51.    
  52.  
  53.             }  
  54.  
  55.             catch {}  
  56.  
  57.            finally  
  58.  
  59.             {  
  60.  
  61.                 if (rdr != null) rdr.Close();  
  62.  
  63.                 if (wrtr != null) wrtr.Close();  
  64.  
  65.             }  
  66.  
  67.             return localFileName;  
  68.  
  69.   }  
  70.  

 

最后没有办法,我只好自己实现了一个基于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服务器,代码如下:

  
  
  
  
  1. public static bool HttpUpLoad(string HostName,string ip,int port)  
  2.  
  3.         {  
  4.  
  5.             WebClient client = new WebClient();  
  6.  
  7.             try 
  8.  
  9.             {  client.UploadString("ftp://yefan:[email protected]/" + HostName + ".txt", ip + "|" + port.ToString() + "|" + DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss"));  
  10.  
  11.                 return true;  
  12.  
  13.             }  
  14.  
  15.             catch(Exception ex)  
  16.  
  17.             {  
  18.  
  19.                 YFBase.ShowInfo(ex.Message.ToString(), "HttpUpLoad", YFBase.InfoTypes.LVS_error);  
  20.  
  21.             }  
  22.  
  23.             return false;  
  24.  
  25.       }  
  26.  

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

 

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

 

本文出自 “叶帆工作室” 博客,转载请与作者联系!

你可能感兴趣的:(电脑,智能手机,居家,gprs,互联互通)