本章通过一个完整的TCP回射(Echo)客户/服务器程序,深入解析TCP套接字编程的核心流程与关键问题。示例程序的功能为:客户端发送文本至服务器,服务器将文本原样返回。通过此案例,读者将掌握:
#include "unp.h"
int main(int argc, char **argv) {
int listenfd, connfd;
pid_t childpid;
socklen_t clilen;
struct sockaddr_in cliaddr, servaddr;
// 创建TCP套接字
listenfd = Socket(AF_INET, SOCK_STREAM, 0);
// 初始化服务器地址结构
bzero(&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY); // 绑定所有接口
servaddr.sin_port = htons(SERV_PORT); // 服务端口号
// 绑定与监听
Bind(listenfd, (SA *)&servaddr, sizeof(servaddr));
Listen(listenfd, LISTENQ); // LISTENQ定义连接队列最大长度
for (;;) {
clilen = sizeof(cliaddr);
connfd = Accept(listenfd, (SA *)&cliaddr, &clilen); // 阻塞等待连接
// 并发处理
if ((childpid = Fork()) == 0) { // 子进程
Close(listenfd); // 子进程关闭监听套接字
str_echo(connfd); // 处理客户端请求
exit(0);
}
Close(connfd); // 父进程关闭已连接套接字
}
}
关键点:
INADDR_ANY
允许服务器监听所有网络接口;fork()
实现并发处理,父进程继续监听新连接,子进程处理当前连接。str_echo
void str_echo(int sockfd) {
ssize_t n;
char buf[MAXLINE];
again:
while ((n = Read(sockfd, buf, MAXLINE)) > 0)
Writen(sockfd, buf, n); // 回射数据
if (n < 0 && errno == EINTR) // 处理中断
goto again;
else if (n < 0)
err_sys("str_echo: read error"); // 包裹函数处理错误
}
注意:TCP是字节流协议,需处理部分读写与粘包问题。
#include "unp.h"
int main(int argc, char **argv) {
int sockfd;
struct sockaddr_in servaddr;
if (argc != 2)
err_quit("usage: tcpcli " );
// 创建TCP套接字
sockfd = Socket(AF_INET, SOCK_STREAM, 0);
// 初始化服务器地址
bzero(&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_port = htons(SERV_PORT);
Inet_pton(AF_INET, argv[1], &servaddr.sin_addr);
// 发起连接
Connect(sockfd, (SA *)&servaddr, sizeof(servaddr));
// 处理用户输入与服务器响应
str_cli(stdin, sockfd);
exit(0);
}
关键点:
Connect
触发三次握手,需处理ETIMEDOUT
(超时)和ECONNREFUSED
(拒绝连接)等错误。str_cli
void str_cli(FILE *fp, int sockfd) {
char sendline[MAXLINE], recvline[MAXLINE];
while (Fgets(sendline, MAXLINE, fp) != NULL) { // 读取标准输入
Writen(sockfd, sendline, strlen(sendline)); // 发送至服务器
if (Readline(sockfd, recvline, MAXLINE) == 0) // 读取响应
err_quit("str_cli: server terminated prematurely");
Fputs(recvline, stdout); // 输出响应
}
}
说明:Readline
需正确处理部分读与缓冲区管理(参考第3章字节流处理)。
wait
,导致进程表中残留条目;void sig_chld(int signo) {
pid_t pid;
int stat;
while ((pid = waitpid(-1, &stat, WNOHANG)) > 0)
printf("child %d terminated\n", pid);
return;
}
// 主函数中注册信号处理
Signal(SIGCHLD, sig_chld); // 使用包裹函数处理信号
作用:捕获SIGCHLD
信号,回收子进程资源。
read
阻塞,TCP持续重传数据,最终返回ETIMEDOUT
;ECONNRESET
错误;read
返回0,触发正常关闭流程;close
释放资源,避免文件描述符泄漏。netstat
监控连接状态netstat -ant | grep 9999 # 查看端口9999的TCP连接状态
输出示例:
LISTEN
:监听状态;ESTABLISHED
:已建立连接;TIME_WAIT
:连接终止等待。tcpdump
抓包分析tcpdump -i lo port 9999 # 监听本地回环接口的9999端口
关键字段:
SYN
/ACK
:三次握手过程;FIN
:四次挥手过程。ps
查看进程状态ps -ef | grep tcpserv # 查看服务器进程状态
状态说明:
S
:睡眠状态(等待I/O);Z
:僵尸进程。pthread_t tid;
pthread_attr_t attr;
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); // 线程分离
pthread_create(&tid, &attr, handle_client, (void *)connfd);
优势:避免频繁创建/销毁线程的开销。
select
/epoll
)小结:本章通过Echo程序完整演示了TCP客户/服务器开发流程,涵盖并发模型、异常处理与调试技巧,为复杂网络应用开发奠定基础。
习题:
Wireshark
分析TCP握手与挥手过程,提交抓包分析报告。付费用户专属资源:
通过本章学习,读者将掌握TCP套接字编程的核心技术,并具备开发高可靠性、高并发网络服务的能力。