C++中TCP/IP套接字编程需创建套接字、绑定地址、监听或连接、收发数据、关闭资源;Windows需初始化WSA,跨平台需处理头文件、关闭函数差异,并注意端口复用、字节序转换与错误处理。

在C++中实现TCP/IP套接字编程,主要依赖操作系统提供的Socket API。Windows和Linux平台的接口略有不同,但基本流程相似:创建套接字、绑定地址、监听(服务端)、连接(客户端)、收发数据、关闭套接字。下面以跨平台思路为基础,分别说明核心步骤与代码实现。
1. TCP服务端的基本实现流程
服务端程序负责监听指定端口,等待客户端连接,并处理通信请求。关键步骤如下:
- 初始化套接字库:Windows需调用WSAStartup,Linux则无需此步。
- 创建套接字:使用socket()函数创建SOCK_STREAM类型的TCP套接字。
- 绑定地址信息:调用bind()将IP地址和端口与套接字关联。
- 监听连接:使用listen()进入等待状态。
- 接受连接:通过accept()获取客户端连接,生成新的通信套接字。
- 数据收发:使用recv()和send()进行通信。
- 关闭资源:通信结束后关闭套接字,Windows需调用WSACleanup()。
示例代码(Windows平台):
#include#include #pragma comment(lib, "ws2_32.lib") int main() { WSADATA wsa; SOCKET server, client; sockaddr_in serv_addr, cli_addr; int addrlen = sizeof(cli_addr);
WSAStartup(MAKEWORD(2,2), &wsa); server = socket(AF_INET, SOCK_STREAM, 0); serv_addr.sin_family = AF_INET; serv_addr.sin_addr.s_addr = INADDR_ANY; serv_addr.sin_port = htons(8080); bind(server, (sockaddr*)&serv_addr, sizeof(serv_addr)); listen(server, 3); std::cout << "等待客户端连接...\n"; client = accept(server, (sockaddr*)&cli_addr, &addrlen); char buffer[1024] = {0}; recv(client, buffer, sizeof(buffer), 0); std::cout zuojiankuohaophpcnzuojiankuohaophpcn "收到消息: " zuojiankuohaophpcnzuojiankuohaophpcn buffer zuojiankuohaophpcnzuojiankuohaophpcn std::endl; const char* reply = "Hello from server"; send(client, reply, strlen(reply), 0); closesocket(client); closesocket(server); WSACleanup(); return 0;}
立即学习“C++免费学习笔记(深入)”;
2. TCP客户端的实现方式
客户端主动连接服务端,发送请求并接收响应。流程更简单:
- 初始化(仅Windows需要)。
- 创建套接字。
- 设置服务端地址(IP和端口)。
- 调用connect()发起连接。
- 使用send()和recv()通信。
- 关闭连接。
客户端示例代码:
#include#include #pragma comment(lib, "ws2_32.lib") int main() { WSADATA wsa; SOCKET sock; sockaddr_in serv_addr;
WSAStartup(MAKEWORD(2,2), &wsa); sock = socket(AF_INET, SOCK_STREAM, 0); serv_addr.sin_family = AF_INET; serv_addr.sin_port = htons(8080); serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1"); if (connect(sock, (sockaddr*)&serv_addr, sizeof(serv_addr)) == 0) { const char* msg = "Hello Server"; send(sock, msg, strlen(msg), 0); char buffer[1024] = {0}; recv(sock, buffer, sizeof(buffer), 0); std::cout zuojiankuohaophpcnzuojiankuohaophpcn "服务器回复: " zuojiankuohaophpcnzuojiankuohaophpcn buffer zuojiankuohaophpcnzuojiankuohaophpcn std::endl; } closesocket(sock); WSACleanup(); return 0;}
立即学习“C++免费学习笔记(深入)”;
3. 跨平台注意事项
Linux与Windows在Socket编程上的主要差异包括:
- 头文件不同:Windows用winsock2.h>,Linux用
、 等。 - 初始化与清理:Windows必须调用WSAStartup()和WSACleanup()。
- 关闭函数:Windows使用closesocket(),Linux使用close()。
- 链接库:Windows需链接ws2_32.lib。
可通过宏定义简化跨平台兼容性:
#ifdef _WIN32 #include#define close closesocket #else #include #include #include #endif
4. 常见问题与建议
实际开发中需要注意以下几点:
- 确保端口未被占用,绑定前可设置SO_REUSEADDR选项避免“Address already in use”错误。
- 所有网络字节序的数据(如端口号)必须使用htons()、htonl()转换。
- 错误检查不可省略,每个关键函数调用后应判断返回值。
- 阻塞模式下recv()可能返回0(对端关闭连接)或-1(出错),需正确处理。
- 多客户端场景下,服务端可用多线程或I/O复用(如select、epoll)提升并发能力。
基本上就这些。掌握基础Socket流程后,可进一步实现非阻塞通信、心跳机制、协议封装等功能。











