1.实验系列
·Linux NAP-Linux网络应用编程系列
2.实验目的
·理解多进程(Multiprocess)相关基本概念,理解父子进程之间的关系与差异,熟练掌握基于fork()的多进程编程模式;
·理解僵尸进程产生原理,能基于|sigaction()或signal(),使用waitpid()规避僵尸进程产生;
·理解Linux 文件系统的组织方式,掌握文件描述符的基本概念,理解主进程 fork()进程后,子进程对于主进程fork()前创建的文件描述符的继承关系;
·在「TCP单进程循环服务器与单进程客户端」的基础上,进一步实践巩固:a.单进程循环服务器套接字编程基本模式;
b.服务器对于客户端正常结束的识别处理;c.客户端基于命令行指令的退出实现方式;
d.服务器基于SIGINT 信号的退出实现方式(僵速系统调用退出问题);
同时,还要进一步理解并掌握TCP多进程并发服务器套接字编程模式与技能,包括:a.多进程并发服务器套接字编程核心系统调用模式:
b.多进程并发服务器规避产生僵尸进程的基本模式(包括 SIGCHLD 处理等);c.简单应用层协议及其PDU的设计、构建与解析处理;
d.文件的读写应用
3.实验内容
·编写TCP多进程循环服务器程序与单进程客户端程序,实现以下主体功能:。客户端启动连接服务器之后,进入命令行交互模式。
操作人员在命令行窗口输入一行字符并回车后,客户端进程立刻从命令行(本质即 stdin)读取数据,并将该行信息发送给服务器。
·服务器收到该行信息后,会将该信息原封不动的返回给客户端,即所谓消息回声(Message Echo)。。客户端收到服务器返回的消息回声后,将其打印输出至屏幕(本质即 stdout)。
·客户端在从命令行收到 EXIT 指令后退出。
·若服务器启动时设定 Established Queue的长度,即listen()第二个参数backlog为2,则最多可 以有2个客户编同时连上服务器并开展交互,此时,再启动另一个客户端连接服务器,观察体验是什么现象,并尝试分析现象背后的底层逻辑。
·本实验不考核以下内容:SIGPIPE 信号处理、基于多次读取的PDU完整获取、PDU 完整设计、多进程客户端.
·本实验不涉及复杂业务,仅要求进行PDU筒单设计(增加了头部要求,但不涉及长度字段),实现简单消息回声服务,以帮助学生理解并构建多进程并发服务器程序的基本框架。
·【重要假设】
·当网络与主机环境均比较理想时,可以支持客户端与服务器实现对于PDU的一次性收发」,即仅通过 read()/write()的一次调用,即可实现PDU(本实验中即消息/消息回声)的「完整收发」.
·本实验中,数据传输量很小(将明确限定一行数据的上限),且测评时客户端与服务器进程均在同一容器内工作,故而不会出现一次收发不能处理「单—PDU」的场景。
服务器端代码:
#include
#include //exit()函数相关
#include //C 和 C++ 程序设计语言中提供对 POSIX 操作系统 API 的访问功能的头文件
#include //Unix/Linux系统的基本系统数据类型的头文件,含有size_t,time_t,pid_t等类型
#include //套接字基本函数相关
#include //IP地址和端口相关定义,比如struct sockaddr_in等
#include //inet_pton()等函数相关
#include //bzero()函数相关
#include
#include
#include //SIGCHLD信号使用
#include
#define BACKLOG 5 //listen函数参数
#define MAXDATASIZE 140
char p[MAXDATASIZE + 100];
void handle_sigint(int sig);
void srv_biz(int connfd,char* veri_code);
void sig_chld(int signo);
void sig_pipe(int signo);
char buf[MAXDATASIZE];
int sigint_flag = 0; // 标记服务器进程是否受到signal信号
int main(int argc, char *argv[]) {
if (argc != 4){ //如果命令行用法不对,则提醒并退出
printf("usage: %s \n",argv[0]);
exit(0);
}
int listenfd, connectfd; //分别是监听套接字和连接套接字
struct sockaddr_in server, client; //存放服务器和客户端的地址信息(前者在bind时指定,后者在accept时得到)
int sin_size; // accept时使用,得到客户端地址大小信息
pid_t pid;
//安装使用SIGPIPE
struct sigaction sigact_pipe;
sigemptyset(&sigact_pipe.sa_mask);
sigact_pipe.sa_handler = sig_pipe;//信号处理函数
sigact_pipe.sa_flags = 0;
sigact_pipe.sa_flags |= SA_RESTART;//设置受影响的慢系
if (sigaction(SIGPIPE, &sigact_pipe, NULL) < 0){
perror("cannot ignore SIGPIPE");
return -1;
}
//安装SIGINT信号处理器
struct sigaction sa;
sa.sa_flags = 0;
sa.sa_handler = handle_sigint;
sigemptyset(&sa.sa_mask);
if(sigaction(SIGINT, &sa, NULL) < 0){
return -1;
}
//注册SIGCHLD的处理函数
struct sigaction sigact_chld, old_sigact_chld;
sigemptyset(&sigact_chld.sa_mask);
sigact_chld.sa_handler = sig_chld;
sigact_chld.sa_flags = 0;
if (sigaction(SIGCHLD, &sigact_chld, &old_sigact_chld) < 0){
return -1;
}
if((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) //建立监听套接字
{
//perror是系统函数,参见https://www.cnblogs.com/noxy/p/11188583.html
perror("Create socket failed.");
exit(-1);
}
int opt = SO_REUSEADDR;
setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)); //将地址和端口设为可立即重用(后续再解释)
//这4行是设置地址结构变量的标准做法,可直接套用
bzero(&server, sizeof(server));
server.sin_family = AF_INET;
server.sin_port = htons(atoi(argv[2]));
if(inet_pton(AF_INET, argv[1], &server.sin_addr) == 0)
{
perror("Server IP Address Error:\n");
exit(1);
}
//把server里的地址信息绑定到监听套接字上
if (bind(listenfd, (struct sockaddr *)&server, sizeof(struct sockaddr)) == -1) {
perror("Bind error.");
exit(-1);
}
if (listen(listenfd, BACKLOG) == -1) { //开始监听
perror("listen error.");
exit(-1);
}
sprintf(p,"[srv](%d)[srv_sa](%s:%s)[vcd](%s) Server has initialized!\n",getpid(),argv[1],argv[2],argv[3]);
fputs(p, stdout);
sin_size = sizeof(struct sockaddr_in);
int sym = 0;
while(!sigint_flag) {
//接受客户端连接(从监听队列里取出)
if ((connectfd = accept(listenfd, (struct sockaddr *)&client, (socklen_t *)&sin_size)) == -1) {
if(errno == EINTR){
continue;
} else{
perror("accept error.");
exit(-1);
}
}
sprintf(p,"[srv] client[%s:%d] is accepted!\n",inet_ntoa(client.sin_addr),client.sin_port);
fputs(p, stdout);
if ((pid= fork()) > 0) { // 父进程
close(connectfd);
continue;
} else if (pid == 0) { // 子进程
close(listenfd);
srv_biz(connectfd, argv[3]);
close(connectfd);
return 0;
} else { // 出现错误
perror("Create child process failed.");
exit(1);
}
close(connectfd); //关闭连接套接字
}
close(listenfd); //关闭监听套接字
}
void srv_biz(int connfd,char* veri_code){
char tep[MAXDATASIZE + 3];
short cid_net;
while(1){
int numbytes; // 从客户端接收字节数
numbytes = read(connfd,&cid_net,2); // 读取2字节的客户端编号
if(numbytes == 0){
break;
}
if((numbytes = read(connfd,buf,MAXDATASIZE)) == -1) {
perror("recv error.");
exit(1);
}
if(numbytes == 0){
break;
}
sprintf(p, "[chd](%d)[cid](%d)[ECH_RQT] %s", getpid(), (short)ntohs(cid_net), buf);
fputs(p,stdout);
bzero(p, sizeof(p));
bzero(tep,sizeof(tep));
short vcd_net = (short)htons((uint16_t)atoi(veri_code));
memcpy(tep, &vcd_net,2);
memcpy(tep + 2, buf, numbytes);
write(connfd, tep, sizeof(tep));
}
}
void handle_sigint(int sig){ // 定义SIGNAL信号处理器
sprintf(p,"[srv] SIGINT is coming!\n");
fputs(p, stdout);
sigint_flag = 1;
}
void sig_chld(int signo){
pid_t pid_chld;
int stat;
while((pid_chld = waitpid(-1, &stat, WNOHANG)) > 0){
sprintf(p, "[srv](%d)[chd](%d) Child has terminated!\n",getppid(),pid_chld);
fputs(p, stdout);
}
}
void sig_pipe(int signo) {
int sig_num = signo;
pid_t pid = getpid();
printf("[srv](%d) SIGPIPE is coming!\n", pid);
}
客户端代码:
#include
#include //exit()函数,atoi()函数
#include //C 和 C++ 程序设计语言中提供对 POSIX 操作系统 API 的访问功能的头文件
#include //Unix/Linux系统的基本系统数据类型的头文件,含有size_t,time_t,pid_t等类型
#include //套接字基本函数
#include //IP地址和端口相关定义,比如struct sockaddr_in等
#include //inet_pton()等函数
#include //bzero()函数
#include
#include
#include
#include
#include
#define MAXDATASIZE 140
char p[MAXDATASIZE + 300];
void handle_sigint(int sig);
void cli_biz(int fd, char* cid);
void sig_chld(int signo);
int sigint_flag = 0; // 标记服务器进程是否受到signal信号
int main(int argc, char *argv[])
{
int clientfd; //clientfd是客户端套接字
struct sockaddr_in server_addr; //存放服务器端地址信息,connect()使用
if (argc != 4){ //如果命令行用法不对,则提醒并退出
printf("usage: %s \n",argv[0]);
exit(0);
}
//安装使用SIGPIPE
struct sigaction ssa;
ssa.sa_handler = SIG_IGN;
sigemptyset(&ssa.sa_mask);
ssa.sa_flags = 0;
if (sigaction(SIGPIPE, &ssa, NULL) < 0){
perror("cannot ignore SIGPIPE");
return -1;
}
//安装SIGINT信号处理器
struct sigaction sa;
sa.sa_flags = 0;
sa.sa_handler = handle_sigint;
sigemptyset(&sa.sa_mask);
sigaction(SIGINT, &sa, NULL);
//注册SIGCHLD的处理函数
struct sigaction sigact_chld, old_sigact_chld;
sigemptyset(&sigact_chld.sa_mask);
sigact_chld.sa_handler = sig_chld;
sigact_chld.sa_flags = 0;
sigaction(SIGCHLD, &sigact_chld, &old_sigact_chld);
if ((clientfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
perror("Create socket failed.");
exit(1);
}
bzero(&server_addr, sizeof(server_addr));
server_addr.sin_family = AF_INET;
// argv[1] 为服务器IP字符串,需要用inet_pton转换为IP地址
if(inet_pton(AF_INET, argv[1], &server_addr.sin_addr) == 0)
{
perror("Server IP Address Error:\n");
exit(1);
}
// argv[2] 为服务器端口,需要用atoi及htons转换
server_addr.sin_port = htons(atoi(argv[2]));
if (connect(clientfd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1) {
perror("connect failed.");
exit(1);
}
sprintf(p,"[cli](%d)[srv_sa](%s:%s) Server is connected!\n",getpid(),argv[1],argv[2]);
fputs(p,stdout);
cli_biz(clientfd, argv[3]);
close(clientfd);
sprintf(p,"[cli] clientfd is closed!\n");
fputs(p, stdout);
sprintf(p,"[cli] clinetfd is to return!\n");
fputs(p, stdout);
return 0;
}
void handle_sigint(int sig){ // 定义SIGNAL信号处理器
sprintf(p,"[srv] SIGINT is coming!\n");
fputs(p, stdout);
sigint_flag = 1;
}
void cli_biz(int clientfd, char* cid){
char buf[MAXDATASIZE + 3]; //缓冲区,用于存放从服务器接收到的信息
char msg[MAXDATASIZE]; // 从命令行中读取的消息
short vcd_net;
while(1){
int numbytes; // numbytes是服务器端接收到的字节数
int msg_len;
fgets(msg,MAXDATASIZE,stdin);
msg_len = strlen(msg);
if(sigint_flag || strncmp(msg, "EXIT", 4) == 0){
sprintf(p,"[cli](%d)[cid](%s)[ECH_RQT] %s",getpid(), cid, msg);
fputs(p,stdout);
bzero(p,sizeof(p));
break;
}
if(msg_len > MAXDATASIZE - 2){
printf("messge is too long!\n");
exit(1);
}
msg[msg_len] = '\0';
sprintf(p,"[cli](%d)[cid](%s)[ECH_RQT] %s",getpid(), cid, msg);
fputs(p,stdout);
bzero(p,sizeof(p));
short cid_net = htons((uint16_t)atoi(cid));
memcpy(buf,&cid_net,2);
memcpy(buf + 2,msg, msg_len);
write(clientfd, buf, sizeof(buf)); //发送原始数据和客户端编号到服务器端
bzero(buf,sizeof(buf));
read(clientfd, &vcd_net, 2); // 读取2字节的验证码
if((numbytes = read(clientfd, buf, MAXDATASIZE)) == -1) {
perror("recv error.");
exit(1);
}
sprintf(p, "[cli](%d)[vcd](%d)[ECH_REP] %s",getpid(), (short)ntohs(vcd_net), buf);
fputs(p,stdout);
bzero(p, sizeof(p));
}
}
void sig_chld(int signo){
pid_t pid_chld;
int stat;
while((pid_chld = waitpid(-1, &stat, WNOHANG)) > 0){
sprintf(p, "[srv](%d)[chd](%d) Child has terminated!\n",getppid(),pid_chld);
fputs(p, stdout);
}
}
注意事项:
关于学生客户端服务器在本地交互测试一切正常,但是上线测试即出现各种错误甚至超时的问题
·在线测试时采用以下模式进行交互:学生客户端<=>标准服务器;标准客户端<=>学生服务器。
·当学生自行编写的客户端、服务器在本地进行交互测试时表现正常,并不能充分说明编码符合题设。
【案例】学生客户端与服务器收发数据时均未进行 PDU字节序转换
学生客户端与服务器收发数据时均未进行PDU字节序转换,本地测试看起来一切正常,但究其根本,是因为学生客户端与服务器程序虽未遵循网络字节序规范,但相当于遵循了无需字节序转换的自定义协议规范,且客户端、服务器进程都运行在同一主机上,因此字节序问题并不会暴露。但标准客户端与标准服务器并不认可该协议,所以上线测试即表现出各种问题。