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

 

GPRS资费受3G的影响逐渐降低,目前5元包月即可以获得30M的流量,而cmwap日渐式微大有被cmnet取代之势(北京GPRS套餐其流量已不区分cmwapcmnet),而后者可无障碍地和因特网互联互通,这种变化更促进了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服务器获取PCIP和端口号;

手机和PC直接通信互联;

实际效果图:

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

探测居家PCIP和端口(从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 respStream = 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:1234@blog.csdn.net/yefanqiu/" + 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程序。(作者:叶帆 200996日)

posted on 2009-09-06 23:02  刘洪峰IoT  阅读(4603)  评论(24编辑  收藏  举报