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 <winsock2.h>
#include <iostream>
#pragma comment(lib, "ws2_32.lib")
<p>int main() {
WSADATA wsa;
SOCKET server, client;
sockaddr_in serv_addr, cli_addr;
int addrlen = sizeof(cli_addr);</p><pre class='brush:php;toolbar:false;'>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 << "收到消息: " << buffer << 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 <winsock2.h>
#include <iostream>
#pragma comment(lib, "ws2_32.lib")
<p>int main() {
WSADATA wsa;
SOCKET sock;
sockaddr_in serv_addr;</p><pre class='brush:php;toolbar:false;'>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 << "服务器回复: " << buffer << std::endl;
}
closesocket(sock);
WSACleanup();
return 0;}
立即学习“C++免费学习笔记(深入)”;
3. 跨平台注意事项
Linux与Windows在Socket编程上的主要差异包括:
- 头文件不同:Windows用<winsock2.h>,Linux用<sys/socket.h>、<netinet/in.h>等。
- 初始化与清理:Windows必须调用WSAStartup()和WSACleanup()。
- 关闭函数:Windows使用closesocket(),Linux使用close()。
- 链接库:Windows需链接ws2_32.lib。
可通过宏定义简化跨平台兼容性:
#ifdef _WIN32 #include <winsock2.h> #define close closesocket #else #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #endif
4. 常见问题与建议
实际开发中需要注意以下几点:
- 确保端口未被占用,绑定前可设置SO_REUSEADDR选项避免“Address already in use”错误。
- 所有网络字节序的数据(如端口号)必须使用htons()、htonl()转换。
- 错误检查不可省略,每个关键函数调用后应判断返回值。
- 阻塞模式下recv()可能返回0(对端关闭连接)或-1(出错),需正确处理。
- 多客户端场景下,服务端可用多线程或I/O复用(如select、epoll)提升并发能力。
基本上就这些。掌握基础Socket流程后,可进一步实现非阻塞通信、心跳机制、协议封装等功能。











