C# Socket编程 同步以及异步通信

转载自:https://www.cnblogs.com/BLoodMaster/archive/2010/07/02/1769774.html

套接字简介:套接字最早是Unix的,window是借鉴过来的。TCP/IP协议族提供三种套接字:流式、数据报式、原始套接字。其中原始套接字允许对底层协议直接访问,一般用于检验新协议或者新设备问题,很少使用。
套接字编程原理:延续文件作用思想,打开-读写-关闭的模式。
C/S编程模式如下:
Ø 服务器端:
打开通信通道,告诉本地机器,愿意在该通道上接受客户请求——监听,等待客户请求——接受请求,创建专用链接进行读写——处理完毕,关闭专用链接——关闭通信通道(当然其中监听到关闭专用链接可以重复循环)
Ø 客户端:打开通信通道,连接服务器——数据交互——关闭信道。
Socket通信方式:
Ø 同步:客户端在发送请求之后必须等到服务器回应之后才可以发送下一条请求。串行运行
Ø 异步:客户端请求之后,不必等到服务器回应之后就可以发送下一条请求。并行运行
套接字模式:
Ø 阻塞:执行此套接字调用时,所有调用函数只有在得到返回结果之后才会返回。在调用结果返回之前,当前进程会被挂起。即此套接字一直被阻塞在网络调用上。
Ø 非阻塞:执行此套接字调用时,调用函数即使得不到得到返回结果也会返回。
套接字工作步骤:
Ø 服务器监听:监听时服务器端套接字并不定位具体客户端套接字,而是处于等待链接的状态,实时监控网络状态
Ø 客户端链接:客户端发出链接请求,要连接的目标是服务器端的套接字。为此客户端套接字必须描述服务器端套接字的服务器地址与端口号。
Ø 链接确认:是指服务器端套接字监听到客户端套接字的链接请求时,它响应客户端链接请求,建立一个新的线程,把服务器端套接字的描述发送给客户端,一旦客户端确认此描述,则链接建立好。而服务器端的套接字继续处于监听状态,继续接受其他客户端套接字请求。
在TCP/IP网络中,IP网络交互分类两大类:面向连接的交互与面向无连接的交互。

Socket构造函数:public socket(AddressFamily 寻址类型, SocketType 套接字类型, ProtocolType 协议类型)。但需要注意的是套接字类型与协议类型并不是可以随便组合。
SocketType
ProtocolType
描述
Stream
Tcp
面向连接
Dgram
Udp
面向无连接
Raw
Icmp
网际消息控制
Raw
Raw
基础传输协议
Socket类的公共属性:
属性名
描述
AddressFamily
获取Socket的地址族
Available
获取已经从网络接收且可供读取的数据量
Blocking
获取或设置一个值,只是socket是否处于阻塞模式
Connected
获取一个值,指示当前连接状态
Handle
获取socket的操作系统句柄
LocalEndPoint
获取本地终端EndPoint
RemoteEndPoint
获取远程终端EndPoint
ProtocolType
获取协议类型
SocketType
获取SocketType类型
Socket常用方法:
Bind(EndPoint)
服务器端套接字需要绑定到特定的终端,客户端也可以先绑定再请求连接
Listen(int)
监听端口,其中parameters表示最大监听数
Accept()
接受客户端链接,并返回一个新的链接,用于处理同客户端的通信问题

Send()
发送数据
Send(byte[])
简单发送数据
Send(byte[],SocketFlag)
使用指定的SocketFlag发送数据
Send(byte[], int, SocketFlag)
使用指定的SocketFlag发送指定长度数据
Send(byte[], int, int, SocketFlag)
使用指定的SocketFlag,将指定字节数的数据发送到已连接的socket(从指定偏移量开始)
Receive()
接受数据
Receive(byte[])
简单接受数据
Receive (byte[],SocketFlag)
使用指定的SocketFlag接受数据
Receive (byte[], int, SocketFlag)
使用指定的SocketFlag接受指定长度数据
Receive (byte[], int, int, SocketFlag)
使用指定的SocketFlag,从绑定的套接字接收指定字节数的数据,并存到指定偏移量位置的缓冲区

Connect(EndPoint)
连接远程服务器
ShutDown(SocketShutDown)
禁用套接字,其中SocketShutDown为枚举,Send禁止发送,Receive为禁止接受,Both为两者都禁止
Close()
关闭套接字,释放资源
异步通信方法:
BeginAccept(AsynscCallBack,object)
开始一个一步操作接受一个连接尝试。参数:一个委托。一个对象。对象包含此请求的状态信息。其中回调方法中必须使用EndAccept方法。应用程序调用BegineAccept方法后,系统会使用单独的线程执行指定的回调方法并在EndAccept上一直处于阻塞状态,直至监测到挂起的链接。EndAccept会返回新的socket对象。供你来同远程主机数据交互。不能使用返回的这个socket接受队列中的任何附加连接。调用BeginAccept当希望原始线程阻塞的时候,请调用WaitHandle.WaitOne方法。当需要原始线程继续执行时请在回调方法中使用ManualResetEvent的set方法
BeginConnect(EndPoint, AsyncCallBack, Object)
回调方法中必须使用EndConnect()方法。Object中存储了连接的详细信息。
BeginSend(byte[], SocketFlag, AsyncCallBack, Object)

BegineReceive(byte[], SocketFlag, AsyncCallBack, Object)

BegineDisconnect(bool, AsyncCallBack, Object)

给出同步通信与异步通信的示例:
同步通信:
预定义结构体,同步通信没有多线程异步委托回调,所以无需预定义结构体
客户端Client:
class Program
{
static void Main()
{
try{
int port = 2000;
string host = “127.0.0.1”;
IPAddress ip = IPAddress.Parse(host);
IPEndPoint ipe = new IPEndPoint(ip, port);//把ip和端口转化为IPEndPoint实例
Socket c = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);//创建一个Socket
Console.WriteLine(“Conneting…”);
c.Connect(ipe);//连接到服务器
string sendStr = “hello!This is a socket test”;
byte[] bs = Encoding.ASCII.GetBytes(sendStr);
Console.WriteLine(“Send Message”);
c.Send(bs, bs.Length, 0);//发送测试信息
string recvStr = “”;
byte[] recvBytes = new byte[1024];
int bytes;
bytes = c.Receive(recvBytes, recvBytes.Length, 0);//从服务器端接受返回信息
recvStr += Encoding.ASCII.GetString(recvBytes, 0, bytes);
Console.WriteLine(“Client Get Message:{0}”, recvStr);//显示服务器返回信息
c.Close();
}
catch (ArgumentNullException e){
Console.WriteLine(“ArgumentNullException: {0}”, e);
}
catch (SocketException e){
Console.WriteLine(“SocketException: {0}”, e);
}
Console.WriteLine(“Press Enter to Exit”);
Console.ReadLine();
}
}
服务器端:
class Program
{
static void Main()
{
try{
int port = 2000;
string host = “127.0.0.1”;
IPAddress ip = IPAddress.Parse(host);
IPEndPoint ipe = new IPEndPoint(ip, port);
Socket s = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);//创建一个Socket类
s.Bind(ipe);//绑定2000端口
s.Listen(0);//开始监听
Console.WriteLine(“Wait for connect”);
Socket temp = s.Accept();//为新建连接创建新的Socket。
Console.WriteLine(“Get a connect”);
string recvStr = “”;
byte[] recvBytes = new byte[1024];
int bytes;
bytes = temp.Receive(recvBytes, recvBytes.Length, 0);//从客户端接受信息
recvStr += Encoding.ASCII.GetString(recvBytes, 0, bytes);
Console.WriteLine(“Server Get Message:{0}”, recvStr);//把客户端传来的信息显示出来
string sendStr = “Ok!Client Send Message Sucessful!”;
byte[] bs = Encoding.ASCII.GetBytes(sendStr);
temp.Send(bs, bs.Length, 0);//返回客户端成功信息
temp.Close();
s.Close();
}
catch (ArgumentNullException e){
Console.WriteLine(“ArgumentNullException: {0}”, e);}
catch (SocketException e){
Console.WriteLine(“SocketException: {0}”, e);}
Console.WriteLine(“Press Enter to Exit”);
Console.ReadLine();
}
}

异步通信:
客户端Client:
预定义结构体,用于异步委托之间的传递。用户根据自己需要定制即可
public class StateObject
{
// Client socket.
public Socket workSocket = null;
// Size of receive buffer.
public const int BufferSize = 256;
// Receive buffer.
public byte[] buffer = new byte[BufferSize];
// Received data string.
public StringBuilder sb = new StringBuilder();
}
正文:
public class AsynchronousClient
{
// The port number for the remote device.
private const int port = 11000;
// ManualResetEvent instances signal completion.
private static ManualResetEvent connectDone = new ManualResetEvent(false);
private static ManualResetEvent sendDone = new ManualResetEvent(false);
private static ManualResetEvent receiveDone = new ManualResetEvent(false);
// The response from the remote device.
private static String response = String.Empty;

private static void StartClient(){
    // Connect to a remote device.
    try{
        // Establish the remote endpoint for the socket.
        // The name of the remote device is "host.contoso.com".
        IPHostEntry ipHostInfo = Dns.Resolve("host.contoso.com");
        IPAddress ipAddress = ipHostInfo.AddressList[0];
        IPEndPoint remoteEP = new IPEndPoint(ipAddress, port);

        // Create a TCP/IP socket.
        Socket client = new Socket(AddressFamily.InterNetwork,
            SocketType.Stream, ProtocolType.Tcp);

        // Connect to the remote endpoint.
        client.BeginConnect(remoteEP,
            new AsyncCallback(ConnectCallback), client);
        connectDone.WaitOne();

        // Send test data to the remote device.
        Send(client, "This is a test<EOF>");
        sendDone.WaitOne();

        // Receive the response from the remote device.
        Receive(client);
        receiveDone.WaitOne();

        // Write the response to the console.
        Console.WriteLine("Response received : {0}", response);

        // Release the socket.
        client.Shutdown(SocketShutdown.Both);
        client.Close();

}
catch (Exception e){
Console.WriteLine(e.ToString());}
}

private static void ConnectCallback(IAsyncResult ar)
{
    try{
        // Retrieve the socket from the state object.
        Socket client = (Socket)ar.AsyncState;

        // Complete the connection.
        client.EndConnect(ar);

        Console.WriteLine("Socket connected to {0}",
            client.RemoteEndPoint.ToString());

        // Signal that the connection has been made.
        connectDone.Set();
    }
    catch (Exception e){
        Console.WriteLine(e.ToString());}
}

private static void Receive(Socket client)
{
    try{
        // Create the state object.
        StateObject state = new StateObject();
        state.workSocket = client;

        // Begin receiving the data from the remote device.
        client.BeginReceive(state.buffer, 0, StateObject.BufferSize, 0,
            new AsyncCallback(ReceiveCallback), state);
    }
    catch (Exception e){
        Console.WriteLine(e.ToString());}
}

private static void ReceiveCallback(IAsyncResult ar)
{
    try{
        // Retrieve the state object and the client socket
        // from the asynchronous state object.
        StateObject state = (StateObject)ar.AsyncState;
        Socket client = state.workSocket;

        // Read data from the remote device.
        int bytesRead = client.EndReceive(ar);

        if (bytesRead > 0){
            // There might be more data, so store the data received so far.
            state.sb.Append(Encoding.ASCII.GetString(state.buffer, 0, bytesRead));

            // Get the rest of the data.
            client.BeginReceive(state.buffer, 0, StateObject.BufferSize, 0,
                new AsyncCallback(ReceiveCallback), state);
        }
        else{
            // All the data has arrived; put it in response.
            if (state.sb.Length > 1)
            {
                response = state.sb.ToString();
            }
            // Signal that all bytes have been received.
            receiveDone.Set();
        }
    }
    catch (Exception e){
        Console.WriteLine(e.ToString());}
}

private static void Send(Socket client, String data)
{
    // Convert the string data to byte data using ASCII encoding.
    byte[] byteData = Encoding.ASCII.GetBytes(data);

    // Begin sending the data to the remote device.
    client.BeginSend(byteData, 0, byteData.Length, 0,
        new AsyncCallback(SendCallback), client);
}

private static void SendCallback(IAsyncResult ar)
{
    try{
        // Retrieve the socket from the state object.
        Socket client = (Socket)ar.AsyncState;

        // Complete sending the data to the remote device.
        int bytesSent = client.EndSend(ar);
        Console.WriteLine("Sent {0} bytes to server.", bytesSent);

        // Signal that all bytes have been sent.
        sendDone.Set();

}
catch (Exception e){
Console.WriteLine(e.ToString());}
}

public static int Main(String[] args)
{
    StartClient();
    return 0;
}

}
服务器端Server:
预定义结构体,用于异步委托之间的传递。同客户端的一致。不再赘述
正文:
// State object for reading client data asynchronously
public class AsynchronousSocketListener
{
// Thread signal.
public static ManualResetEvent allDone = new ManualResetEvent(false);
public AsynchronousSocketListener(){}
public static void StartListening()
{
// Data buffer for incoming data.
byte[] bytes = new Byte[1024];
// Establish the local endpoint for the socket.
// The DNS name of the computer
// running the listener is “host.contoso.com”.

    //IPHostEntry ipHostInfo = Dns.Resolve(Dns.GetHostName());
    IPHostEntry ipHostInfo = Dns.Resolve("127.0.0.1");
    IPAddress ipAddress = ipHostInfo.AddressList[0];
    IPEndPoint localEndPoint = new IPEndPoint(ipAddress, 11000);
    // Create a TCP/IP socket.
    Socket listener = new Socket(AddressFamily.InterNetwork,SocketType.Stream, ProtocolType.Tcp);
    // Bind the socket to the local endpoint and listen for incoming connections.
    try{
        listener.Bind(localEndPoint);
        listener.Listen(100);
        while (true){
            // Set the event to nonsignaled state.
            allDone.Reset();
            // Start an asynchronous socket to listen for connections.
            Console.WriteLine("Waiting for a connection...");
            listener.BeginAccept(new AsyncCallback(AcceptCallback),listener);
            // Wait until a connection is made before continuing.
            allDone.WaitOne();
        }
    }
    catch (Exception e){
        Console.WriteLine(e.ToString());}
    Console.WriteLine("\nPress ENTER to continue...");
    Console.Read();
}
public static void AcceptCallback(IAsyncResult ar)
{
    // Signal the main thread to continue.
    allDone.Set();
    // Get the socket that handles the client request.
    Socket listener = (Socket)ar.AsyncState;
    Socket handler = listener.EndAccept(ar);
    // Create the state object.
    StateObject state = new StateObject();
    state.workSocket = handler;
    handler.BeginReceive(state.buffer, 0, StateObject.BufferSize, 0,new AsyncCallback(ReadCallback), state);
}
public static void ReadCallback(IAsyncResult ar)
{
    String content = String.Empty;
    // Retrieve the state object and the handler socket
    // from the asynchronous state object.
    StateObject state = (StateObject)ar.AsyncState;
    Socket handler = state.workSocket;
    // Read data from the client socket.
    int bytesRead = handler.EndReceive(ar);
    if (bytesRead > 0)
    {
        // There    might be more data, so store the data received so far.
        state.sb.Append(Encoding.ASCII.GetString(state.buffer, 0, bytesRead));
        // Check for end-of-file tag. If it is not there, read
        // more data.
        content = state.sb.ToString();
        if (content.IndexOf("<EOF>") > -1){
            // All the data has been read from the
            // client. Display it on the console.
            Console.WriteLine("Read {0} bytes from socket. \n Data : {1}",
            content.Length, content);
            // Echo the data back to the client.
            Send(handler, "Server return :" + content);
        }
        else{
            // Not all data received. Get more.
            handler.BeginReceive(state.buffer, 0, StateObject.BufferSize, 0,
            new AsyncCallback(ReadCallback), state);
        }
    }
}
private static void Send(Socket handler, String data){
    // Convert the string data to byte data using ASCII encoding.
    byte[] byteData = Encoding.ASCII.GetBytes(data);
    // Begin sending the data to the remote device.
    handler.BeginSend(byteData, 0, byteData.Length, 0,
    new AsyncCallback(SendCallback), handler);
}
private static void SendCallback(IAsyncResult ar)
{
    try{
        // Retrieve the socket from the state object.
        Socket handler = (Socket)ar.AsyncState;
        // Complete sending the data to the remote device.
        int bytesSent = handler.EndSend(ar);
        Console.WriteLine("Sent {0} bytes to client.", bytesSent);
        handler.Shutdown(SocketShutdown.Both);
        handler.Close();
    }
    catch (Exception e){
        Console.WriteLine(e.ToString());
    }
}
public static int Main(String[] args)
{
    StartListening();
    return 0;
}

}

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在VS2019使用C#进行Socket异步通信的步骤如下: 1. 创建一个新的C#控制台应用程序项目。 2. 在项目添加一个新的类文件,用于处理Socket通信的逻辑。 3. 在类文件引入System.Net.Sockets命名空间,以便使用Socket相关的类和方法。 4. 创建一个Socket对象,并设置其属性和参数,如IP地址、端口号、协议类型等。 5. 使用Socket对象的BeginConnect方法发起异步连接请求,传入服务器的IP地址和端口号作为参数。 6. 创建一个回调函数,用于处理连接请求的结果。回调函数的参数是IAsyncResult类型,可以通过它获取连接状态和错误信息。 7. 在回调函数,使用EndConnect方法结束异步连接请求,并检查连接是否成功建立。 8. 如果连接成功建立,可以开始进行数据的发送和接收操作。 9. 使用Socket对象的BeginSend和BeginReceive方法发起异步发送和接收请求,传入要发送或接收的数据作为参数。 10. 创建两个回调函数,分别用于处理发送和接收操作的结果。回调函数的参数是IAsyncResult类型,可以通过它获取发送或接收的状态和错误信息。 11. 在发送和接收的回调函数,使用EndSend和EndReceive方法结束异步发送和接收请求,并处理发送或接收到的数据。 12. 可以根据需要重复执行步骤9到11,实现连续的数据发送和接收。 13. 当不再需要进行通信时,使用Socket对象的Close方法关闭Socket连接。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值