.net平台下C#socket通信(转)
上篇.net平台下C#socket通信(上)介绍了socket通信的基本原理及最基本的通信方式。本文在此基础上就socket通信时经常遇到的问题做一个简单总结,都是项目中的一些小问题,拿来此处便于下次使用,同时对在使用socket时出现些许问题的同仁们多一个粗浅建议。不足之处请提出,谢谢。
本文主要讲述:
1、正常通信中握手建立
2、一对多的通信
3、发送接收数据格式转换
4、资源释放
5、开启并保持服务监听
1、握手建立正常的通信通道
项目需要通信的双方(假设是一个上位机、一个下位机)之间需要建立一个稳定的通道,以便进行通信。本项目中具体操作是:上位机作为服务器,下位机作为客户端,同时制定通信协议。上位机首先打开监听等待建立通道,下位机主动连接上位机后发送连接成功的信息到上位机,上位机根据通信协议发送数据到下位机,此时通道已经建立。但为了保险起见(同时遵循三次握手),客户端再次发送数据到上位机告知通道建立完毕。
2、一对多通信
项目需求是一个上位机多个下位机,这就确定了上位机做为服务器端,下位机作为客户端主动连接服务器。一对一通信时只有一个socket通道,因此无论是上位机还是下位机在发送和接收数据的时候都不会存在数据乱发乱收的情况。一对多意味着上位机和下位机会建立起多个通道,因此在发送数据时需要记录哪一个下位机处于哪个socket通道中,以便进行逻辑处理。本文处理一对多通信的过程是:
1)首先建立一个对话类Session:

public class Session
{
public Socket ClientSocket { get; set; }//客户端的socket
public string IP;//客户端的ip public Session(Socket clientSocket)
{
this.ClientSocket = clientSocket;
this.IP = GetIPString();
} public string GetIPString()
{
string result = ((IPEndPoint)ClientSocket.RemoteEndPoint).Address.ToString();
return result;
}
}

2)在服务端socket监听时:

IPEndPoint loaclEndPoint = new IPEndPoint(IPAddress.Any, Port);
SocketLister = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
SocketLister.Bind(loaclEndPoint); try
{
SocketLister.Listen(MaxConnection);
while (IsRunning)
{
ClientSocket = SocketLister.Accept();
//保存socket
Session newSession = new Session(ClientSocket);
lock (sessionLock)
{
sessionTable.Add(newSession.IP, newSession);
} SocketConnection socketConnection = new SocketConnection(ClientSocket);
socketConnection.ReceiveDatagram();//接收数据
}
}
catch (SocketException ex)
{
} //声明 public Hashtable sessionTable = new Hashtable ();//包括客户端会话
private object sessionLock = new object();

为了便于理解,把整个服务端socket的建立都写在上面。
3)发送数据到不同的客户端

Hashtable ht = serverSocket.sessionTable;
foreach (Session session in ht.Values)
{
if (session.IP == "127.0.0.1")//example
{
SocketConnection socketConnection = new SocketConnection(session.ClientSocket);
string str = "C300010002D2";
byte[] sendBytes = StrToHexByte(str);
socketConnection.Send(sendBytes);
}
}

SocketConnection类已经被使用多次,写在下面:

public class SocketConnection:IDisposable
{
public ServerSocket Server { get; set; }
public Byte[] MsgBuffer = null;
private int totalLength = 0;
public int CurrentBufferLength; private Socket _ClientSocket = null;
public Socket ClientSock
{
get{ return this._ClientSocket; }
} public SocketConnectionType Type { get; private set; } #region Constructor
public SocketConnection(ServerSocket server, Socket sock)
{
this.Server = server;
this._ClientSocket = sock;
this.Type = SocketConnectionType.Server;
} public SocketConnection(Socket sock)
{
this._ClientSocket = sock;
this.Type = SocketConnectionType.Client;
}
#endregion #region Events
public SocketConnectionDelegate OnConnect = null;//是否连接
public SocketConnectionDelegate OnLostConnect = null;//中断连接
public ReceiveDataDelegate OnReceiveData = null;//接收数据 #endregion #region Connect
public void Connect(IPAddress ip, int port)
{
this.ClientSock.BeginConnect(ip, port, ConnectCallback, this.ClientSock);
} private void ConnectCallback(IAsyncResult ar)
{
try
{
Socket handler = (Socket)ar.AsyncState;
handler.EndConnect(ar);
if (OnConnect != null)
{
OnConnect(this);
}
ReceiveDatagram();
}
catch (SocketException ex)
{
}
}
#endregion #region Send
public void Send(string data)
{
Send(System.Text.Encoding.UTF8.GetBytes(data));
} public void Send(byte[] byteData)
{
try
{
int length = byteData.Length;
byte[] head = BitConverter.GetBytes(length);
byte[] data = new byte[head.Length + byteData.Length];
Array.Copy(head, data, head.Length);
Array.Copy(byteData, 0, data, head.Length, byteData.Length);
this.ClientSock.BeginSend(data, 0, data.Length, 0, new AsyncCallback(SendCallback), this.ClientSock);
}
catch (SocketException ex)
{
}
} private void SendCallback(IAsyncResult ar)
{
try
{
Socket handler = (Socket)ar.AsyncState;
handler.EndSend(ar);
}
catch (SocketException ex)
{
}
} #endregion #region ReceiveDatagram
public void ReceiveDatagram()
{
SocketStateObject state = new SocketStateObject();
state.workSocket = _ClientSocket;
_ClientSocket.BeginReceive(state.buffer, 0, SocketStateObject.BufferSize, 0, new AsyncCallback(ReceiveCallback), state);
} private void ReceiveCallback(IAsyncResult ar)
{
SocketStateObject state = (SocketStateObject)ar.AsyncState;
Socket handler = state.workSocket;
if (handler.Connected)
{
try
{
state.bytesRead = handler.EndReceive(ar);
if (state.bytesRead > 0)
{
OnDataRecivedCallback(state.buffer, state.bytesRead);
Array.Clear(state.buffer, 0, state.buffer.Length);
state.bytesRead = 0;
handler.BeginReceive(state.buffer, 0, SocketStateObject.BufferSize, 0, new AsyncCallback(ReceiveCallback), state);
}
else
{
//if (OnDisconnect != null)
//{
// OnDisconnect(this);
//}
Dispose();
}
}
catch (SocketException ex)
{ }
}
} private void ReceiveCallBack00(IAsyncResult ar)
{
try
{
int REnd = _ClientSocket.EndReceive(ar);
if (REnd > 0)
{ OnDataRecivedCallback(MsgBuffer, REnd );
Array.Clear(MsgBuffer, 0, MsgBuffer.Length);
REnd = 0; _ClientSocket.BeginReceive(MsgBuffer, 0, MsgBuffer.Length, 0, new AsyncCallback(ReceiveCallBack00), null);
}
else
{
if (OnLostConnect != null)
{
OnLostConnect(this);
} Dispose();
}
}
catch (Exception ex)
{ } } private void OnDataRecivedCallback(byte[] data, int length)
{
if (length > 0)
{
if (this.MsgBuffer == null)
{
byte[] bytelength = new byte[4];
Array.Copy(data, bytelength, 4);
this.totalLength = BitConverter.ToInt32(bytelength, 0);
this.MsgBuffer = new byte[this.totalLength];
this.CurrentBufferLength = 0;
}
if (this.totalLength > 0)
{
int offset = 0;
if (CurrentBufferLength == 0)
{
offset = 4;
}
if (length + this.CurrentBufferLength >= this.totalLength + offset)
{
int len = this.totalLength - CurrentBufferLength;
Array.Copy(data, offset, this.MsgBuffer, this.CurrentBufferLength, len);
byte[] tmp = this.MsgBuffer.Clone() as byte[];
OnReceiveData(new MessageData(this, tmp));
this.MsgBuffer = null;
if (length - len - offset > 0)
{
tmp = new byte[length - len - offset];
Array.Copy(data, offset + len, tmp, 0, tmp.Length);
OnDataRecivedCallback(tmp, tmp.Length);
}
}
else
{
Array.Copy(data, offset, this.MsgBuffer, this.CurrentBufferLength, length - offset);
this.CurrentBufferLength += length - offset;
}
}
else
{
}
}
} public void Dispose()
{
try
{
this.ClientSock.Shutdown(SocketShutdown.Both);
this.ClientSock.Close();
this.Server = null; //if (OnLostConnect != null)
//{
// OnLostConnect(this);
//}
}
catch
{
}
}
#endregion
}

3、处理需要发送和接收到的数据
项目需要是上位机获取数据进行逻辑处理,然后通过tcp/ip协议发送给下位机,下位机在接收到数据的同时发送确认信息到上位机。项目过程中,上位机在开发过程中需要调试其发送数据、接收数据是否成功,此处借助于USR- TCP232小工具。但是涉及到一个问题,下位机发送和接收都是byte字节数组,那么开发的上位机应该如何发送和接收数据?在.net平台下C#socket通信(上),有服务器端的发送和接收函数,发送数据时将要发送的字符串转换为byte数组,接收时再将字节数组转换为16进制字符串。如下:

//字节数组转换为16进制字符串
public string ByteToHexStr(byte[] bytes)
{
string str = "";
if (bytes != null)
{
for (int i = 0; i < bytes.Length; i++)
{
str += bytes[i].ToString("X2");
}
} return str;
} //字符串转换为16进制byte数组
public byte[] StrToHexByte(string data)
{
data = data.Replace(" ", "");
if ((data.Length % 2) != 0)
{
data += " ";
} byte[] bytes = new byte[data.Length / 2];
for (int i = 0; i < bytes.Length; i++)
{
bytes[i] = Convert .ToByte (data.Substring (i * 2,2),16);
} return bytes;
}

4、资源释放
开发项目使用平台是.net,工具vs2010,语言是C#,因为.net有垃圾回收机制,因此在实际开发中产生的托管资源都是系统自动释放完成。在做本项目时采用winform进行开发的,在此过程中发现一个问题:在关闭Form窗体是运行的系统并没有完全关闭。查找原因,应该是有资源没有被释放。而socket套接字产生的资源恰好是非托管资源,此现象表明系统中有socket资源没有被完全释放掉。因此写了一个资源释放函数:

public void Dispose()
{
try
{
this.ClientSocket.Shutdown(SocketShutdown.Both);
this.ClientSocket.Dispose();
this.ClientSocket.Close();
this.ClientSocket = null;
}
catch
{
}
}

上述函数的功能就是释放socket所产生的资源,调用后发现还是存在此问题,几经调试发现虽然把产生socket通道的监听客户端资源释放完毕,服务器端的serversocket并没有被释放,于是有了下一个函数:

public void CloseSocket()
{
if (serverSocket != null)
{
serverSocket.SocketLister.Dispose();
serverSocket.SocketLister = null;
serverSocket.Dispose();//调用的上一个函数
serverSocket = null;
}
}

在上述函数完成后,套接字socket所产生的资源确实被释放完毕,系统在form关闭后能真正关闭。到此资源好像已经被释放掉,但紧接着新的问题产生了:
在什么时候什么地方调用释放资源的函数?
个人简单看法:
1)系统中止时调用
2)socket通道中断时调用
补充:
5、开启并保持服务监听
在socket通信中,服务端的socket监听其实是需要一直打开并且保持的,只有这样才能随时监听连接的客户端。项目中示例:

private void button1_Click(object sender, RoutedEventArgs e)
{
Thread thread = new Thread(new ThreadStart(StartServer));
thread.Start();
} public void StartServer()
{
int port = Convert.ToInt32(GetText(this.tbPort));
string ipStr = GetText (this.tbServerIPStr);
if (serverSocket == null)
{
serverSocket = new ServerSocket(port);
serverSocket.Start(ipStr);//
}
else
{
MessageBox.Show("监听已开启");
}
} public void Start(string ipStr)
{
IPEndPoint localEndPoint = new IPEndPoint(IPAddress.Any, Port);
//IPEndPoint localEndPoint = new IPEndPoint(IPAddress.Parse(ipStr), Port);
SocketLister = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
SocketLister.Bind(localEndPoint); try
{
SocketLister.Listen(MaxConnection);
while (IsRunning)
{
ClientSocket = SocketLister.Accept();
//保存socket
Session newSession = new Session(ClientSocket);
lock (sessionLock)
{
sessionTable.Add(newSession.IP, newSession);
} SocketConnection socketConnection = new SocketConnection(ClientSocket);
socketConnection.ReceiveDatagram();
}
}
catch (SocketException ex)
{
}
}

解释:点击按钮开启新的线程thread,执行方法StartServer,StartServer调用方法Start,Start方法中使用死循环开启并保持监听。
.net平台下C#socket通信(转)的更多相关文章
- .net平台下C#socket通信(中)
上篇.net平台下C#socket通信(上)介绍了socket通信的基本原理及最基本的通信方式.本文在此基础上就socket通信时经常遇到的问题做一个简单总结,都是项目中的一些小问题,拿来此处便于下次 ...
- .net平台下C#socket通信(上)
在开始介绍socket前先补充补充基础知识,在此基础上理解网络通信才会顺理成章,当然有基础的可以跳过去了.都是废话,进入正题. TCP/IP:Transmission Control Protocol ...
- .net 平台下, Socket通讯协议中间件设计思路(附源码)
.net 平台下,实现通讯处理有很多方法(见下表),各有利弊: 序号 实现方式 特点 1 WCF 优点:封装好,方便.缺点:难学,不跨平台 2 RocketMQ,SuperSocket等中间件 优点: ...
- 移动平台下的Socket几个问题
在页游时代,使用Flash ActionScript 3.0进行开发,as3提供比较简单和健全的socket API.到了手游时代,基于tcp的socket编程遇到了一些棘手的问题.通常情况下手游都要 ...
- 一步一步从原理跟我学邮件收取及发送 4.不同平台下的socket
既然是面向程序员的文章那当然不能只说说原理,一定要有实际动手的操作. 其实作为我个人的经历来说,对于网络编程,这是最重要的一章! 作为一位混迹业内近20年的快退休的程序员,我学习过很多的开发语言 ...
- C#socket通信1
.net平台下C#socket通信(上) 完全是基础,新手可以随意看看,大牛可以关闭浏览页了,哈哈. 在开始介绍socket前先补充补充基础知识,在此基础上理解网络通信才会顺理成章,当然有基础的可以跳 ...
- .net平台下socket异步通讯(代码实例)
你应该知道的.net平台下socket异步通讯(代码实例) 1,首先添加两个windows窗体项目,一个作为服务端server,一个作为客户端Client 2,然后添加服务端代码,添加命名空间,界面上 ...
- Android平台下的TCP/IP传输(客户端)
在工科类项目中,嵌入式系统与软件系统或后台数据库之间的信息传输是实现“物联网”的一种必要的途径,对已简单概念的物联网,通常形式都是一个单片机/嵌入式系统实现数据的采集及其处理,通过蓝牙,wifi或者是 ...
- 基于TCP与UDP协议的socket通信
基于TCP与UDP协议的socket通信 C/S架构与初识socket 在开始socket介绍之前,得先知道一个Client端/服务端架构,也就是 C/S 架构,互联网中处处充满了 C/S 架构(Cl ...
随机推荐
- bjfu1100 圆环
这题也是2011百度之星的一道题.知道做法后代码极简单. 不过我做完后随便上网搜了一下,发现竟然还有很多不同的做法.别的做法我就不管了,我只把我的做法的原理说清楚.我做题时是按如下顺序逐步找到规律的: ...
- Web自动化框架搭建——前言
1.web测试功能特性 a.功能逻辑测试(功能测试),这一块所有系统都是一致的,比如数据的添加.删除.修改:功能测试案例设计感兴趣和有时间的话可以另外专题探讨: b.浏览器兼容性测试,更重要的是体验这 ...
- linux 常用命令基础
linux常用的命令 shell 是命令语句,命令解释程序以及程序设计语言的统称,它不仅仅拥有自己内建的shell命令集,同时也能被系统中其他应用程序所调用 shell 的一个重要特性是它本身就是一个 ...
- php环境配置中各个模块在网站建设中的功能
上一篇配置环境的时候,我们注意到,有四个模块需要配置,那么,这四个模块分别有哪些功能呢? 一.php php是我们的用来创建动态网页的强有力的脚本语言,安装过程中我们直接解压到某一个路径就好了,比 ...
- append some buttons to the standard datagrid pager bar
<script type="text/javascript"> $(function(){ var pager = $('#dg').datagrid('getP ...
- [iOS基础控件 - 6.12.4] NavigationController vs TabBarController
A.属性 1. Item NavigationController: navigationItem (不需要创建) title rightBarButtonItems/ rightBarButtonI ...
- Lua学习笔记(四):表和数组
表 在Lua中,表(table)是十分重要的一种数据结构,实际上Lua对于复杂数据类型也仅提供了表给我们,我们通过表可以实现我们需要的大部分重要的数据结构,比如数组. table类型实现了关联数组,关 ...
- Objective-C的singleton模式
最近因为在ios应用开发中,考虑到一些公共方法的封装使用,就决定使用单例模式的写法了..不知道,Object-c中的单例模式的写法是否和java中的写法是否有所区别?于是阿堂从网上一搜,发现“ Obj ...
- PC/UVa 题号: 110106/10033 Interpreter (解释器)题解 c语言版
, '\n'); #include<cstdio> #include<iostream> #include<string> #include<algorith ...
- PL/pgSQL学习笔记之二
39.1.1 使用 PL/pgSQL的好处 SQL是 PostgreSQL和其他大多数关系型数据库作为查询语言而使用的语言.它可移植,并容易学习.但是SQL语句必须被数据库服务器逐条地执行. 这意味着 ...