C#的Socket编程细节

目录

Socket中的Accept

步骤1:创建并绑定服务端套接字

步骤2:接受连接请求

步骤3:与客户端通信

步骤4:关闭套接字

注意事项

Socket中的Connected

使用Connected属性

客户端检查连接状态

服务端检查连接状态

注意事项

Socket中的RemoteEndPoint

使用RemoteEndPoint属性

服务端获取客户端的端点信息

客户端获取服务器的端点信息

注意事项


Socket中的Accept

在C#的Socket编程中,Accept方法是服务端套接字用来接受客户端的连接请求的。当服务端套接字监听到客户端的连接请求时,它会调用Accept方法来创建一个新的套接字,这个新的套接字用于与客户端进行通信。

以下是使用Accept方法的步骤:

步骤1:创建并绑定服务端套接字

cs 复制代码
// 创建套接字
Socket listener = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

// 绑定到本地IP和端口
IPAddress localIP = IPAddress.Any; // 监听所有网络接口
int port = 11000; // 端口号
IPEndPoint localEndPoint = new IPEndPoint(localIP, port);

// 绑定到指定的端口和IP
listener.Bind(localEndPoint);

// 开始监听
listener.Listen(100); // 可以同时挂起的连接请求的最大数量

步骤2:接受连接请求

cs 复制代码
// 接受连接请求,这将阻塞直到一个客户端连接
Socket clientSocket = listener.Accept();

// 连接被接受后,clientSocket用于与客户端通信
// 可以进行发送和接收数据的操作

步骤3:与客户端通信

一旦接受连接,就可以使用返回的SocketclientSocket)来发送和接收数据。

cs 复制代码
// 接收数据
byte[] buffer = new byte[1024];
int bytesRead = clientSocket.Receive(buffer);

// 发送数据
string message = "Hello from server";
byte[] data = Encoding.ASCII.GetBytes(message);
clientSocket.Send(data);

步骤4:关闭套接字

完成通信后,应该关闭套接字。

cs 复制代码
// 关闭客户端套接字
clientSocket.Shutdown(SocketShutdown.Both);
clientSocket.Close();

// 在适当的时候也关闭监听套接字
listener.Shutdown(SocketShutdown.Both);
listener.Close();

注意事项

  • Accept方法是阻塞的,也就是说,它会暂停执行,直到一个客户端连接到服务端。
  • 如果你希望在等待连接时不阻塞主线程,可以使用异步版本的方法,如BeginAcceptEndAccept
  • Accept方法返回一个新的Socket对象,用于与客户端进行通信,而原始的监听套接字(listener)继续监听新的连接请求。
  • 在高并发场景下,通常需要在单独的线程或使用异步方式处理每个接受的连接,以避免服务器因等待I/O操作而变得无响应。

使用Accept方法时,确保你的代码能够妥善处理异常和错误情况,例如客户端在通信过程中突然断开连接。

Socket中的Connected

在C#的Socket类中,Connected属性用于检查套接字是否已连接到远程资源。这个属性返回一个布尔值,指示套接字是否已建立连接。

使用Connected属性

以下是如何使用Connected属性的示例:

客户端检查连接状态
cs 复制代码
// 创建套接字
Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

try
{
    // 连接到服务端
    socket.Connect(new IPEndPoint(IPAddress.Parse("127.0.0.1"), 11000));
    
    // 检查是否已连接
    if (socket.Connected)
    {
        Console.WriteLine("Connected to the server.");
        // 可以进行数据发送和接收操作
    }
    else
    {
        Console.WriteLine("Failed to connect to the server.");
    }
}
catch (Exception e)
{
    Console.WriteLine("Exception: " + e.Message);
}
finally
{
    // 关闭套接字
    socket.Shutdown(SocketShutdown.Both);
    socket.Close();
}
服务端检查连接状态

服务端通常不直接使用Connected属性,因为它是通过监听和接受客户端连接来建立通信的。但是,如果你在服务端需要检查与特定客户端的连接状态,也可以使用Connected属性。

cs 复制代码
// 接受连接
Socket clientSocket = listener.Accept();

// 检查是否已连接
if (clientSocket.Connected)
{
    Console.WriteLine("Client connected.");
    // 可以进行数据发送和接收操作
}
else
{
    Console.WriteLine("Client not connected.");
}

注意事项

  • Connected属性仅指示套接字是否已连接,它不会告诉你连接的质量或网络延迟。如果需要检测连接的活跃状态,你可能需要定期发送心跳包或使用其他协议特定的机制。
  • 如果连接已经断开,Connected属性可能不会立即返回false。这是因为TCP的断开过程是四次挥手,这个过程可能需要一些时间来完成。
  • 在异步操作中,你应该在捕获异常后检查Connected属性,以确定是否需要重新连接或处理错误。
  • 在多线程环境中,如果多个线程共享同一个套接字,Connected属性的值可能会在多个线程之间竞争。确保在访问Connected属性时进行适当的同步。

Socket中的RemoteEndPoint

在C#的Socket类中,RemoteEndPoint属性用于获取与套接字连接的远程端点的EndPoint。这个属性在你想要获取连接到你的服务端套接字的客户端的IP地址和端口号时非常有用,或者当你需要知道一个客户端套接字连接到的服务器的端点信息时。

使用RemoteEndPoint属性

以下是如何使用RemoteEndPoint属性的示例:

服务端获取客户端的端点信息
cs 复制代码
// 创建套接字并监听
Socket listener = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
IPEndPoint localEndPoint = new IPEndPoint(IPAddress.Any, 11000);
listener.Bind(localEndPoint);
listener.Listen(100);

// 接受连接
Socket clientSocket = listener.Accept();
Console.WriteLine("Client connected.");

// 获取远程端点信息
if (clientSocket.Connected)
{
    IPEndPoint remoteEndPoint = (IPEndPoint)clientSocket.RemoteEndPoint;
    Console.WriteLine("Connected to client: {0}", remoteEndPoint.Address.ToString());
}
客户端获取服务器的端点信息

客户端也可以使用RemoteEndPoint来获取它连接到的服务器的端点信息。

cs 复制代码
// 创建套接字并连接到服务器
Socket clientSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
IPEndPoint serverEndPoint = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 11000);
clientSocket.Connect(serverEndPoint);

// 获取远程端点信息
if (clientSocket.Connected)
{
    IPEndPoint remoteEndPoint = (IPEndPoint)clientSocket.RemoteEndPoint;
    Console.WriteLine("Connected to server: {0}", remoteEndPoint.Address.ToString());
}

注意事项

  • RemoteEndPoint在套接字成功连接后才能正确返回远程端点信息。
  • 如果套接字尚未连接,尝试访问RemoteEndPoint可能会抛出异常。
  • RemoteEndPoint通常用于调试目的,或者在需要动态处理来自不同客户端的连接时。
  • 在异步操作中,你应该在确保套接字连接成功后再访问RemoteEndPoint属性。
  • RemoteEndPoint返回的类型是EndPoint,如果你知道它一定是IPEndPoint类型,可以进行适当的类型转换。
相关推荐
ZWZhangYu3 分钟前
声明式LoggerFactory.getLogger和注解@Slf4j原理区别和推荐
java·开发语言
CodeLinghu3 分钟前
SpringBoot高级并发实践:自定义线程池与@Async异步调用深度解析
java·spring boot·后端
我与岁月的森林17 分钟前
游戏中的对象池技术探索(一)
unity·c#·对象池
今天不coding1 小时前
自定义持久层框架------从零到一手写一个mybatis
java·mybatis
Android技术栈1 小时前
鸿蒙开发(NEXT/API 12)【蓝牙服务开发】网络篇
网络·华为·蓝牙·harmonyos·鸿蒙·鸿蒙系统·openharmony
码农阿岩~1 小时前
适合初学者的[JAVA]: 服务框架常见问题
java·开发语言
Beginner_bml2 小时前
TCP协议的三次握手与四次挥手的过程
网络·网络协议·tcp/ip
小羊在奋斗2 小时前
【C++】二叉搜索树+变身 = AVL树
java·开发语言·c++·机器学习
疯一样的码农3 小时前
Spring Boot Starter Parent介绍
java·spring boot·后端
GEEKVIP3 小时前
选择 PDF 编辑器时要考虑什么?如何选择适用于 Windows 10 的 PDF 编辑器
网络·windows·ios·智能手机·pdf·编辑器·iphone