Linux客户/服务器程序设计范式1——并发服务器(多进程)
引言
本文会写一个并发服务器(concurrent server)程序,它为每个客户请求fork出一个子进程。
注意
1. 信号处理问题
对于相同信号,按信号的先后顺序依次处理。可能会产生的问题是,正在处理sig1信号时,又来了2个或更多的sig1信号,此sig1时只会在处理完原来的sig1信号后,再处理1个sig1信号。因此对于相同信号,会产生信号掉包的问题。 一个儿子退了之后,程序在处理handler(),如果此时又退了两个儿子,那么必然有一个儿子的资源回收不到,称为僵尸进程。
对于不同信号,优先处理后者,处理完后者在回头处理上一个。例如正在处理sig1时,来了sig2,则会先处理sig2,等处理完sig2后,回过头继续处理sig1。
2. 解决方案
在注册的信号处理函数中,使用循环,并在循环中用waitpid来回收子进程资源。只要进入信号处理函数,那么该信号处理函数就可以把所有fork出的儿子的资源都回收掉。注意:waitpid要设置成非阻塞模式,不然当进入循环后,如果没有子进程退出时,会阻塞在信号处理函数中。
3. wait与waitpid
wait一定是阻塞模式。因此在信号处理函数中,用while1{wait(NULL)}有问题,如果进入信号处理函数后,只要有儿子不退,就会一直阻塞在这里。
waitpid可以是阻塞模式,也可以是非阻塞模式。
4. select与accept
两者在收到信号时,均会返回-1。
对于系统默认不处理的信号,程序收不到,两者当然也不会返回-1。换句话说,SIGCHLD是系统默认不处理的信号,如果不对其注册信号处理函数,当子进程退出时,select与accept也是不会返回-1的。
5. 本代码用到了Linux网络编程9——对TCP与UDP的简易封装2.0中的动态库。
代码
server.c
/*************************************************************************
> File Name: server.c
> Author: KrisChou
> Mail:zhoujx0219@163.com
> Created Time: Fri 05 Sep 2014 03:31:12 PM CST
************************************************************************/
#include "my_socket.h"
#include <sys/wait.h>
#include <signal.h>
#include <errno.h>
#define MY_IP "192.168.1.100"
#define MY_PORT 8888
#define SIZE 192
#define MSG_SIZE (SIZE - 4)
extern int errno ;
typedef struct tag_mag
{
int msg_len;//记录msg_buf的真实大小
char msg_buf[MSG_SIZE];//msg_buf所占空间为188byte
}MSG, *pMSG; void my_handle(int num)
{
/*waitpid参数:
* -1表示回收每一个儿子
* NULL表示不关心子进程的exit的返回值
* WNOHANG:wait no hang 非阻塞模式
*waitpid返回值:
* -1表示没有创建任何儿子
* 0表示没有儿子退出
*大于0表示有儿子退出
* */
while(waitpid(-1, NULL, WNOHANG ) > 0) ;
} int main(int argc, char* argv[])
{
int fd_listen , fd_client ;
signal(SIGCHLD, my_handle);
my_socket(&fd_listen, MY_TCP, MY_IP ,MY_PORT);
my_listen(fd_listen, 10); while( fd_client = accept(fd_listen, NULL, NULL))
{
/* 只要不是程序默认忽略的信号,accept都能收到,并返回-1 */
if(fd_client == -1)
{
if(errno == EINTR)
{
continue ;
}else
{
break ; //break退出后,父亲就退了。下来儿子会由init接管。
}
}else
{
if(fork() == 0) //fork儿子用于与客户端通信
{
MSG recv_msg ;
int recvn;
while(1 )
{
memset(&recv_msg, 0, sizeof(MSG));
/*在my_socket.c中,my_recv接收的长度 与 my_send 发送的长度必须是精确值
* my_recv中填的长度小于等于实际要收的,是可以的,大于的话就永远退不出循环了*/
my_recv(&recvn, fd_client, &recv_msg, 4);
if(recvn == 0) //当对面客户端退出(关闭socket),系统调用recv的返回值为0
{
break ;
}else
{
my_recv(NULL,fd_client, &recv_msg.msg_buf, recv_msg.msg_len);
my_send(NULL, fd_client, &recv_msg, 4 + recv_msg.msg_len); }
}
close(fd_client);
exit(0);
}
close(fd_client);
}
}
return 0 ;
}
client.c
#include "my_socket.h"
#define MY_IP "192.168.1.100"
#define MY_PORT 6666
#define SER_IP "192.168.1.100"
#define SER_PORT 8888
#define SIZE 192
#define MSG_SIZE (SIZE - 4)
typedef struct tag_mag
{
int msg_len ;
char msg_buf[MSG_SIZE];//188
}MSG, *pMSG;
int main(int argc, char* argv[])
{
int sfd ;
my_socket(&sfd, MY_TCP, MY_IP, MY_PORT);
my_connect(sfd, SER_IP, SER_PORT);
MSG my_msg ;
while(memset(&my_msg, 0, sizeof(MSG)), fgets(my_msg.msg_buf, MSG_SIZE, stdin)!= NULL)
{
my_msg.msg_len = strlen(my_msg.msg_buf);
my_send(NULL, sfd, &my_msg, 4 + my_msg.msg_len );
memset(&my_msg, 0, sizeof(MSG));
my_recv(NULL, sfd, &my_msg, 4);
my_recv(NULL, sfd, &my_msg.msg_buf, my_msg.msg_len);
printf("recv from server : %s \n", my_msg.msg_buf); }
close(sfd); }
注意:本代码由于client.c中绑定了端口,因此只能连一个客户端,读者可以自己从命令行中输入端口号或者让系统自行分配。
本范式的缺陷在于,服务端在每次收到一个客户端连接请求后,才会fork儿子进行处理,fork有时间开销。更好的方法是,服务端提前fork好儿子,下篇博文会讲进程池。
Linux客户/服务器程序设计范式1——并发服务器(多进程)的更多相关文章
- Linux客户/服务器程序设计范式2——并发服务器(进程池)
引言 让服务器在启动阶段调用fork创建一个子进程池,通过子进程来处理客户端请求.子进程与父进程之间使用socketpair进行通信(为了方便使用sendmsg与recvmsg,如果使用匿名管道,则无 ...
- UNP学习笔记(第三十章 客户/服务器程序设计范式)
TCP测试用客户程序 #include "unp.h" #define MAXN 16384 /* max # bytes to request from server */ in ...
- Linux网络编程服务器模型选择之并发服务器(下)
前面两篇文章(参见)分别介绍了循环服务器和简单的并发服务器网络模型,我们已经知道循环服务器模型效率较低,同一时刻只能为一个客户端提供服务,而且对于TCP模型来说,还存在单客户端长久独占与服务器的连接, ...
- Linux C++服务器程序设计范式
<Unix网络编程>30章详细介绍了几种服务器设计范式.总结了其中的几种,记录一下: 多进程的做法: 1.每次创建一个新的请求,fork一个子进程,处理该连接的数据传输. 2.预先派生一定 ...
- Linux网络编程服务器模型选择之并发服务器(上)
与循环服务器的串行处理不同,并发服务器对服务请求并发处理.循环服务器只能够一个一个的处理客户端的请求,显然效率很低.并发服务器通过建立多个子进程来实现对请求的并发处理.并发服务器的一个难点是如何确定子 ...
- UNIX网络编程卷1 server程序设计范式1 并发server,为每一个客户请求fork一个进程
本文为senlie原创.转载请保留此地址:http://blog.csdn.net/zhengsenlie 1.传统并发server调用 fork 派生一个子进程来处理每一个客户 2.传统并发serv ...
- Linux网络编程客户\服务器设计范式
1.前言 网络编程分为客户端和服务端,服务器通常分为迭代服务器和并发服务器.并发服务器可以根据多进程或多线程进行细分,给每个连接创建一个独立的进程或线程,或者预先分配好多个进程或线程等待连接的请求.今 ...
- LINUX环境并发服务器的三种实现模型
服务器设计技术有很多,按使用的协议来分有TCP服务器和UDP服务器.按处理方式来分有循环服务器和并发服务器. 1 循环服务器与并发服务器模型 在网络程序里面,一般来说都是许多客户对应一个服务器,为了 ...
- linux学习之高并发服务器篇(二)
高并发服务器 1.线程池并发服务器 两种模型: 预先创建阻塞于accept多线程,使用互斥锁上锁保护accept(减少了每次创建线程的开销) 预先创建多线程,由主线程调用accept 线程池 3.多路 ...
随机推荐
- Android操作系统11种传感器介绍
我们依次看看这十一种传感器 1 加速度传感器 加速度传感器又叫G-sensor,返回x.y.z三轴的加速度数值. 该数值包含地心引力的影响,单位是m/s^2. 将手机平放在桌面上,x轴默认为0,y轴默 ...
- UIWebView swift
// // ViewController.swift // UILabelTest // // Created by mac on 15/6/23. // Copyright (c) 2015年 fa ...
- Java Day 14
多线程--线程间通信 对同一个资源进行处理,但是任务却不同 线程间通信--等待唤醒机制 1.wait(); 线程处于冻结状态,被wait线程存储在线程池中 2.notify(); 从线程池唤醒一个 ...
- [vsftp服务]——ftp虚拟用户、权限设置等的实验
搭建ftp服务器,满足以下要求: 1.允许匿名用户登录服务器并下载文件,下载速度设置为最高2MB/s 2.不允许本地用户登录ftp服务器 3.在服务器添加虚拟用户vuser01.vuser02.vus ...
- android开发,socket发送文件,read阻塞,得不到文件尾-1
这是我的接收文件代码:开始可以读取到-1,但是现在又读取不到了,所以才加上红色字解决的(注释的代码) File file = new File(mfilePath,"chetou." ...
- httphelp web自动化
public class HttpHelper { public static CookieContainer CookieContainers = new CookieConta ...
- 基于AutoCAD的空间数据共享平台雏形
好久没有更新博客了,今天先透露一个新的产品——AutoMap.我自己对于这个产品的定位是“基于AutoCAD的空间数据共享平台”.用一句话来概括AutoMap的功能:为用户提供一个在AutoCAD下访 ...
- 【BZOJ】【3856】Monster
又是一道水题…… 重点是分情况讨论: 首先我们很容易想到,如果a*k-b*(k+1)>0的话那么一定能磨死Monster. 但即使不满足这个条件,还有可能打死boss: 1.h-a<1也就 ...
- android控件---自定义带文本的ImageButton
由于SDK提供的ImageButton只能添加图片,不能添加文字:而Button控件添加的文字只能显示在图片内部:当我们需要添加文字在图片外部时就不能满足我们的需求了,顾只能自己写个自定义ImageB ...
- 土地购买 usaco 斜率优化
看这道题的时候,感觉很难,因为数据范围比较大,很难dp: 后来想到了[书柜的尺寸]这道题,也是一道dp,曾经看了那道题的题解而深有启发: 这道题每组的付费只与这一组长宽的最大值有关,也就是说要分组,一 ...