.NET平台下C# socket通信

.NET平台下C# socket通信

https://blog.csdn.net/subin_iecas/article/details/80289915

 

在开始介绍socket前先补充补充基础知识,在此基础上理解网络通信才会顺理成章。

TCP/IP:Transmission Control Protocol/Internet Protocol,传输控制协议/因特网互联协议,又名网络通讯协议。

简单来说:TCP控制传输数据,负责发现传输的问题,一旦有问题就发出信号,要求重新传输,直到所有数据安全正确地传输到目的地,而IP是负责给因特网中的每一台电脑定义一个地址,以便传输。从协议分层模型方面来讲:TCP/IP由:网络接口层(链路层)、网络层、传输层、应用层。它和OSI的七层结构以及对于协议族不同,下图简单表示:

 

 

现阶段socket通信使用TCP、UDP协议,相对应UDP来说,TCP则是比较安全稳定的协议了。本文只涉及到TCP协议来说socket通信。一般建立TCP需要三次握手才能建立,而断开连接则需要四次握手。整个过程如下图所示,在握手基础上延伸socket通信的基本过程。

 

表1 TCP/IP结构

 

 

图1 TCP/IP关系图

 

 

图2 三次握手 四次握手关系

 

 

在此基础上,socket连接过程:

服务器监听:服务器端socket并不定位具体的客户端socket,而是处于等待监听状态,实时监控网络状态。

客户端请求:客户端clientSocket发送连接请求,目标是服务器的serverSocket。为此,clientSocket必须知道serverSocket的地址和端口号,进行扫描发出连接请求。

连接确认:当服务器socket监听到或者是受到客户端socket的连接请求时,服务器就响应客户端的请求,建议一个新的socket,把服务器socket发送给客户端,一旦客户端确认连接,则连接建立。

注:在连接确认阶段:服务器socket即使在和一个客户端socket建立连接后,还在处于监听状态,仍然可以接收到其他客户端的连接请求,这也是一对多产生的原因。

下图简单说明连接过程:

 

 

代码可以详见,附件中的内容,不过唯一值得说一下的事情是,其实在TCP/IP传输的数据都应该是以字节为单位的。比如说传送50个double类型的数据就是传送400个字节的数组。所以在这个过程中,我们首先需要将这一类的数据首先转化成为字节数组才能进行传递。在这个过程中,LabVIEW主要是通过一个节点完成的转化。这个函数的主要的操作就是将任意类型的数据转化为字节数组然后在进行数据传输。

 

 

这里又要进行一个说明,由于C#本身的原因,所以在字节存储格式的时候都是由小端进行存储的,但是在TCP/IP传输格式的时候标准默认的时候都是用大端方式进行传输的,所以拿到的数据不能直接进行解析。由于这个原因这边编写一个可以完成大小端转化的类,方便用户在TCP/IP以及其他串口等类型的时候进行使用。其中GetBytes可以完成多种类型的转化,包括数值和数组。使用十分方便。

 

 

下图所示是范例运行的过程,其中127.0.0.1,是使用本机IP号的时候IP地址,其中一个是Server端一个是Client端。

 

 

同时与LabVIEW Simple TCP的范例可以共同使用,相互做Server与Client都没有问题。如下图所示。

 

 

介绍了通信的过程以及机制,但实际上这中间简单的TCP的通信在实际应用中是比较的,利用C# TCP多线程的应用案例,这边一起来分析一下多线程的代码,大家也可以在文章附带的Demo进行尝试。

我们点击启动服务按钮,服务器:

 

            // 创建负责监听的套接字,注意其中的参数;

            socketWatch = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

            // 获得文本框中的IP对象;

            IPAddress address = IPAddress.Parse(txtIp.Text.Trim());

                // 创建包含ip和端口号的网络节点对象;

                IPEndPoint endPoint = new IPEndPoint(address, int.Parse(txtPort.Text.Trim()));

                try

                {

                    // 将负责监听的套接字绑定到唯一的ip和端口上;

                    socketWatch.Bind(endPoint);

                }

                catch (SocketException se)

                {

                    MessageBox.Show("异常:"+se.Message);

                    return;

                }

 

 

首先我们创建负责监听的套接字, 在Bind绑定后,我们创建了负责监听的线程。代码如下:

 

             // 设置监听队列的长度;

            socketWatch.Listen(10);

            // 创建负责监听的线程;

            threadWatch = new Thread(WatchConnecting);

            threadWatch.IsBackground = true;

            threadWatch.Start();

            ShowMsg("服务器启动监听成功!");

            btnBeginListen.Enabled = false;

 

其中 WatchConnecting方法是负责监听新客户端请求的。然后让我们看一下WatchConnecting的代码。

 

        /// <summary>

        /// 监听客户端请求的方法;

        /// </summary>

        void WatchConnecting()

        {

            while (true)  // 持续不断的监听客户端的连接请求;

            {

                // 开始监听客户端连接请求,Accept方法会阻断当前的线程;

                Socket sokConnection = socketWatch.Accept(); // 一旦监听到一个客户端的请求,就返回一个与该客户端通信的 套接字;

                // 想列表控件中添加客户端的IP信息;

                lbOnline.Items.Add(sokConnection.RemoteEndPoint.ToString());

                // 将与客户端连接的 套接字 对象添加到集合中;

                dict.Add(sokConnection.RemoteEndPoint.ToString(), sokConnection);

                ShowMsg("客户端连接成功!");

                Thread thr = new Thread(RecMsg);

                thr.IsBackground = true;

                thr.Start(sokConnection);

                dictThread.Add(sokConnection.RemoteEndPoint.ToString(), thr);  //  将新建的线程 添加 到线程的集合中去。

            }

        }

 

这个线程是一直存在的,主要的任务就是监听是否有Client与Server端进行连接,如果连接成功则会另开一个线程”RecMsg”。在该线程中则主要是得到字符数据的处理,包括接受数据以及发送数据。

 

void RecMsg(object sokConnectionparn)

        {

                Socket sokClient = sokConnectionparn as Socket;

                while (true)

                {

                    // 定义一个2M的缓存区;

                    byte[] arrMsgRec = new byte[1024 * 1024 * 2];

                    // 将接受到的数据存入到输入  arrMsgRec中;

                    int length = -1;

                    try

                    {

                        length = sokClient.Receive(arrMsgRec); // 接收数据,并返回数据的长度;

                    }

                    catch (SocketException se)

                    {

                        ShowMsg("异常:" + se.Message);

                        // 从 通信套接字 集合中删除被中断连接的通信套接字;

                        dict.Remove(sokClient.RemoteEndPoint.ToString());

                        // 从通信线程集合中删除被中断连接的通信线程对象;

                        dictThread.Remove(sokClient.RemoteEndPoint.ToString());

                        // 从列表中移除被中断的连接IP

                        lbOnline.Items.Remove(sokClient.RemoteEndPoint.ToString());

                        break;

                    }

                    catch (Exception e)

                    {

                        ShowMsg("异常:" + e.Message);

                        // 从 通信套接字 集合中删除被中断连接的通信套接字;

                        dict.Remove(sokClient.RemoteEndPoint.ToString());

                        // 从通信线程集合中删除被中断连接的通信线程对象;

                        dictThread.Remove(sokClient.RemoteEndPoint.ToString());

                        // 从列表中移除被中断的连接IP

                        lbOnline.Items.Remove(sokClient.RemoteEndPoint.ToString());

                        break;

                    }

                    if (arrMsgRec[0] == 0)  // 表示接收到的是数据;

                    {

                        string strMsg = System.Text.Encoding.UTF8.GetString(arrMsgRec,1, length-1);// 将接受到的字节数据转化成字符串;

                        ShowMsg(strMsg);

                    }

                    if (arrMsgRec[0] == 1) // 表示接收到的是文件;

                    {

                            SaveFileDialog sfd = new SaveFileDialog();

                          

                            if (sfd.ShowDialog(this) == System.Windows.Forms.DialogResult.OK)

                            {// 在上边的 sfd.ShowDialog() 的括号里边一定要加上 this 否则就不会弹出 另存为 的对话框,而弹出的是本类的其他窗口,,这个一定要注意!!!【解释:加了this的sfd.ShowDialog(this),“另存为”窗口的指针才能被SaveFileDialog的对象调用,若不加thisSaveFileDialog 的对象调用的是本类的其他窗口了,当然不弹出“另存为”窗口。】

                              

                                string fileSavePath = sfd.FileName;// 获得文件保存的路径;

                                // 创建文件流,然后根据路径创建文件;

                                using (FileStream fs = new FileStream(fileSavePath, FileMode.Create))

                                {

                                    fs.Write(arrMsgRec, 1, length - 1);

                                    ShowMsg("文件保存成功:" + fileSavePath);

                                }

                            }

                        }

                }    

        }

 

其实这就是建立多线程TCP通信的主要过程,这里值得注意的就是其实监听线程监听的一直都是一个固定的端口,在应用层如果建立建立连接了,则连接不会使用监听的端口号,而会使用另一个空闲的端口号,这样才能保证一直连接监听一直使用一个固定端口号,从而使得连接也变得更加容易。Demo有数据交换的,也有类似于通信的,大家都可以参考。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章