在上一篇程序框架中,解决了子进程退出,父进程继续存在的功能,但是多条客户端连接如果同一时间并行退出,
导致服务器端多个子进程同一时间全部退出,而SIGCHLD是不可靠信号,同时来多条信号可能无法处理,导致出现僵尸进程,
如果使用while循环wait又会阻塞父进程,这里采取waitpid()函数来解决这个问题。
//辅助类实现
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <sys/wait.h>
#include <signal.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include "pub.h" ssize_t readn(int fd, const void *buf, ssize_t count)
{
if (buf == NULL)
{
printf("readn() params is not correct !\n");
return -;
}
//定义剩余字节数
ssize_t lread = count;
//定义辅助指针变量
char *pbuf = (char *) buf;
//定义每次读取的字节数
ssize_t nread = ;
while (lread > )
{
nread = read(fd, pbuf, lread);
if (nread == -)
{
//read是可中断睡眠函数,需要屏蔽信号
if (errno == EINTR)
continue;
perror("read() err");
return -;
} else if (nread == )
{
printf("peer read socket is closed !\n");
//返回已经读取的字节数
return count - lread;
}
//重置剩余字节数
lread -= nread;
//辅助指针后移
pbuf += nread;
}
return count;
} ssize_t writen(int fd, const void *buf, ssize_t count)
{
if (buf == NULL)
{
printf("writen() params is not correct !\n");
return -;
}
//定于剩余字节数
ssize_t lwrite = count;
//定义每次写入字节数
ssize_t nwrite = ;
//定义辅助指针变量
char *pbuf = (char *) buf;
while (lwrite > )
{
nwrite = write(fd, pbuf, lwrite);
if (nwrite == -)
{
if (errno == EINTR)
continue;
perror("write() err");
return -;
} else if (nwrite == )
{
printf("peer write socket is closed !\n");
return count - lwrite;
}
//重置剩余字节数
lwrite -= nwrite;
//辅助指针变量后移
pbuf += nwrite;
}
return count;
} ssize_t recv_peek(int fd, const void *buf, ssize_t count)
{
if (buf == NULL)
{
printf("recv_peek() params is not correct !\n");
return -;
}
ssize_t ret = ;
while ()
{
//此处有多少读取多少,不一定ret==count
ret = recv(fd, (void *) buf, count, MSG_PEEK);
if (ret == - && errno == EINTR)
continue;
return ret;
}
return -;
} ssize_t mreadline(int fd, const void *buf, ssize_t count)
{
//定义剩余字节数
ssize_t lread = count;
//定义每次读取的字节数
ssize_t nread = ;
//定义辅助指针变量
char *pbuf = (char *) buf;
int i = , ret = ;
while ()
{
nread = recv_peek(fd, pbuf, count);
if (nread == -)
{
perror("recv_peek() err");
return -;
} else if (nread == )
{
//注意:这里一个客户端有两个进程,也就是套接字会关闭两次,会向服务器发送两次FIN信号
printf("peer socket is closed !\n");
return -;
}
for (i = ; i < nread; i++)
{
if (pbuf[i] == '\n')
{
//这是一段报文
memset(pbuf, , count);
//从socket缓存区读取i+1个字节
ret = readn(fd, pbuf, i + );
if (ret != i + )
return -;
return ret;
}
}
//如果当前socket缓存区中没有\n,
//那么先判断自定义buf是否还有空间,如果没有空间,直接退出
//如果有空间,先将当前socket缓存区中的数据读出来,放入buf中,清空socket缓存
//继续recv,判断下一段报文有没有\n
if (lread >= count)
{
printf("自定义buf太小了!\n");
return -;
}
//读取当前socket缓存
ret = readn(fd, pbuf, nread);
if (ret != nread)
return -;
lread -= nread;
pbuf += nread;
}
return -;
} void handler(int sign)
{
if (sign == SIGCHLD)
{
int mypid=;
//WNOHANG 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若结束,则返回该子进程的ID。
while((mypid=waitpid(-,NULL,WNOHANG))>)
{
printf("子进程pid=%d\n",mypid);
}
//wait(NULL);
}
} int server_socket()
{
int listenfd = socket(AF_INET, SOCK_STREAM, );
if (listenfd == -)
{
perror("socket() err");
return -;
}
//reuseaddr
int optval = ;
if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval))
== -)
{
perror("setsockopt() err");
return -;
}
//bind
struct sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons();
addr.sin_addr.s_addr = inet_addr("127.0.0.1");
if (bind(listenfd, (struct sockaddr *) &addr, sizeof(addr)) == -)
{
perror("bind() err");
return -;
}
//listen
if (listen(listenfd, SOMAXCONN) == -)
{
perror("listen()err");
return -;
}
pid_t pid = ;
//忽略SIGCHLD信号
//signal(SIGCHLD,SIG_IGN);
//安装信号
if (signal(SIGCHLD, handler) == SIG_ERR)
{
printf("signal() failed !\n");
return -;
}
while ()
{
struct sockaddr_in peeraddr;
socklen_t peerlen = sizeof(peeraddr);
int conn = accept(listenfd, (struct sockaddr *) &peeraddr, &peerlen);
printf("accept by %s\n", inet_ntoa(peeraddr.sin_addr));
if (conn == -)
{
perror("accept() err");
return -;
}
pid = fork();
if (pid == -)
{
perror("fork() err");
return -;
}
//子进程接收数据
if (pid == )
{
//关闭监听套接字
close(listenfd);
char buf[] = { };
int ret = ;
while ()
{
ret = mreadline(conn, buf, );
if (ret == -)
{
close(conn);
return -;
}
//打印客户端数据
fputs(buf, stdout);
//把数据返回给客户端
writen(conn, buf, ret);
memset(buf, , sizeof(buf));
}
} else if (pid > )
{
close(conn);
}
}
return ;
} int client_say(int fd)
{
int ret = ;
char buf[] = { };
while (fgets(buf, , stdin) != NULL)
{
//发送数据
ret = writen(fd, buf, strlen(buf));
if (ret != strlen(buf))
return -;
memset(buf, , sizeof(buf));
ret = mreadline(fd, buf, sizeof(buf));
if (ret == -)
{
return -;
}
fputs(buf,stdout);
memset(buf, , sizeof(buf));
}
return ;
} int client_socket()
{
int sockarr[] = { };
int i = ;
//同时创建5个连接
for (i = ; i < ; i++)
{
sockarr[i] = socket(AF_INET, SOCK_STREAM, );
if (sockarr[i] == -)
{
perror("socket() err");
return -;
}
//bind
struct sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons();
addr.sin_addr.s_addr = inet_addr("127.0.0.1");
if (connect(sockarr[i], (struct sockaddr *) &addr, sizeof(addr)) == -)
{
perror("connect() err");
return -;
}
//获取本机地址
struct sockaddr_in myaddr;
socklen_t mylen = sizeof(myaddr);
if (getsockname(sockarr[i], (struct sockaddr *) &myaddr, &mylen) == -)
{
perror("getsockname() err");
return -;
}
printf("本次连接地址:%s\n",inet_ntoa(myaddr.sin_addr));
}
client_say(sockarr[]);
return ;
}

Linux 网络编程详解七(并发僵尸进程处理)的更多相关文章

  1. TCP/UDP Linux网络编程详解

    本文主要记录TCP/UDP网络编程的基础知识,采用TCP/UDP实现宿主机和目标机之间的网络通信. 内容目录 1. 目标2.Linux网络编程基础2.1 嵌套字2.2 端口2.3 网络地址2.3.1 ...

  2. Linux 网络编程详解六(多进程服务器僵尸进程解决方案)

    小结:在点对点p2p程序中,服务器端子程序退出,子进程会主动发送信号,关闭父进程,但是这种模式导致服务器只能支持一个客户端连接,本章节中使用新的框架,子进程退出,不主动发送信号关闭父进程,而是父进程安 ...

  3. Linux 网络编程详解九

    TCP/IP协议中SIGPIPE信号产生原因 .假设客户端socket套接字close(),会给服务器发送字节段FIN: .服务器接收到FIN,但是没有调用close(),因为socket有缓存区,所 ...

  4. Linux 网络编程详解二(socket创建流程、多进程版)

    netstat -na | grep " --查看TCP/IP协议连接状态 //socket编程提高版--服务器 #include <stdio.h> #include < ...

  5. Linux 网络编程详解一(IP套接字结构体、网络字节序,地址转换函数)

    IPv4套接字地址结构 struct sockaddr_in { uint8_t sinlen;(4个字节) sa_family_t sin_family;(4个字节) in_port_t sin_p ...

  6. Linux 网络编程详解十一

    /** * read_timeout - 读超时检测函数,不含读操作 * @fd:文件描述符 * @wait_seconds:等待超时秒数,如果为0表示不检测超时 * 成功返回0,失败返回-1,超时返 ...

  7. Linux 网络编程详解十

    select int select(int nfds, fd_set *readfds, fd_set *writefds,fd_set *exceptfds, struct timeval *tim ...

  8. Linux 网络编程详解五(TCP/IP协议粘包解决方案二)

    ssize_t recv(int s, void *buf, size_t len, int flags); --与read相比,只能用于网络套接字文件描述符 --当flags参数的值设置为MSG_P ...

  9. Linux 网络编程详解四(流协议与粘包)

    TCP/IP协议是一种流协议,流协议是字节流,只有开始和结束,包与包之间没有边界,所以容易产生粘包,但是不会丢包. UDP/IP协议是数据报,有边界,不存在粘包,但是可能丢包. 产生粘包问题的原因 . ...

随机推荐

  1. Hierarchyid 常用操作

        ---------内置函数------------ select hierarchyid::GetRoot()--0x select hierarchyid::Parse('/1/1/') - ...

  2. 在SQL2008中使用XML应对不确定结构的参数

    目的:统一接口,当数据结构发生变化时,前后端业务接口不发生变化,由业务具体解析结构. 规则:确定的接口用参数表(多行提交),不确定的参数用XML   DECLARE @r TABLE     (    ...

  3. OSGI.NET 框架浅析

    关于osgi.net ,想必大家也听说过,以下是自己在学习osgi.net 过程中整理出来的内容,供大家学习参与使用. 1.  UIOSP 开放工厂框架架构 开放工厂所有插件基于OSGi.NET面向服 ...

  4. Linux账号密码过期会导致crontab作业不能执行

    今天一同事报告Linux服务器上的crontab作业没有运行,检查/var/log/cron日志后发现下面错误信息 Jan 19 16:30:01 xxxx crond[31399]: Authent ...

  5. CentOS7安装图形界面和修改运行级别

    CentOS7系统如果用mini镜像安装或者服务器版本安装,默认是没有安装图形界面的.如果需要额外去安装图形界面,可以手动来安装CentOS Gnome GUI包.然后会总结一下,在CentOS7系统 ...

  6. Java 注释说明

    注释 什么是注释呢?就是标注解释的意思,主要用来对Java代码进行说明.Java中有三种注释方式 (1):// :注释单行语句 示例: //定义一个值为10的int变量 int a = 10; (2) ...

  7. 地图编辑器V3

    V3.2.4 (2014-07-03) ---------------------------1. 保存地图的锁定与可视状态:2. 地图单独存为map格式结尾的文件与导出的XML文件区别:3. 修正瓷 ...

  8. 使用Tcmalloc进行堆栈分析

    在前一篇译文<使用TCmalloc的堆栈检查>,介绍了Tcmalloc进行堆栈检查,今天翻译<heap-profiling using tcmalloc>,了解如何 TCmal ...

  9. 清除MAC OS X上的流氓软件 - advance mac cleaner

    自3721开天辟地以来,流氓软件从来就没有消停过,连MAC OS X都难逃流氓软件的骚扰. 近日,因为从SourceForge上下载了一个软件安装包,结果中招了——莫名其妙被安装了advance ma ...

  10. MMORPG大型游戏设计与开发(客户端架构 part12 of vegine)

    在游戏中的交互过程中输入是一个必不可少的过程,比如登陆的时候需要用户输入用户名与密码,就算是单机游戏很多时候也要求用户输入一个用户名作为存档的依据.网络游戏中没有了输入,只用鼠标来交互是不切实际的,因 ...