进程概念及应用

我们知道,监听套接字会有一个等待队列,里面存放着不同客户端的连接请求,如果有一百个客户端,每个客户端的请求处理是0.5s,第一个客户端当然不会不满,但第一百个客户端就会有相当大的意见了。为了要使得所有客户端都尽可能的满意,我们应采用并发服务端,使其同时向所有发起请求的客户端提供服务。而且,网络程序中数据通信时间比CPU运算时间占比更大,因此,向多个客户端提供服务是一种有效利用CPU的方式。接下来讨论同时向多个客户端提供服务的并发服务端,下面提出具有代表性的并发服务端实现模型和方法:

  • 多进程服务器:通过创建多个进程提供服务
  • 多路复用服务器:通过捆绑并统一管理I/O对象提供服务
  • 多线程服务器:通过生成与客户端等量的线程提供服务

先来简单理解下进程:我们打开电脑一般不会只做一件事,比方单纯的浏览网站,单纯的聊天。一般我们都是几件事轮流切换着做,我们会在浏览网页时打开音乐播放器播放音乐,还会时不时回复下QQ消息。那么这里就牵扯到三个进程了,一个是浏览器进程,一个是播放器进程,还有一个是QQ进程。从操作系统的角度看,进程是程序流的基本单位,若创建多个进程,则操作系统将同时运行。有时一个程序运行过程中也会产生多个进程,像谷歌浏览器,打开一个tab页,实际上就是产生一个新的进程。接下来要创建的多进程服务器就是其中的代表,编写服务端前,先了解一下通过程序创建进程的方法

CPU核的个数和进程数:拥有两个运算器的CPU称为双核CPU,拥有四个运算器的CPU称作四核CPU。也就是说,一个CPU可能包含多个运算器(核)。核的个数与可同时运行的进程数相同,相反,若进程数超过核数,进程将分时使用CPU资源。但因CPU运算速度极快,我们会感到所有进程同时运行,当然,核数越多,这种感觉越明显

进程ID

讲解创建进程方法前,先简要说明下进程ID。无论进程是如何创建的,所有进程都会从操作系统分配得到ID。此ID称为“进程ID”,其值为大于2的整数,1要分配给操作系统启动后的(用于协助操作系统)首个进程,因此用于进程无法得到ID为1的进程ID,接下来观察Linux中正在运行的进程:

  1. # ps au
  2. USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
  3. root 384 0.0 0.0 1520 208 pts/23 Ss+ Sep04 0:00 /bin/sh -c nginx -g "daemon on;" && uwsgi --ini /data/web/uwsgi.ini
  4. root 438 0.0 0.6 257212 36936 pts/23 Sl+ Sep04 0:03 uwsgi --ini /data/web/uwsgi.ini
  5. root 473 0.0 0.0 1520 208 pts/3 Ss+ Sep21 0:00 /bin/sh -c nginx -g "daemon on;" && uwsgi --ini /data/web/uwsgi.ini
  6. root 513 0.0 0.7 186080 44028 pts/3 S+ Sep21 0:05 uwsgi --ini /data/web/uwsgi.ini
  7. root 555 0.0 0.6 186724 40404 pts/3 Sl+ Sep21 0:00 uwsgi --ini /data/web/uwsgi.ini
  8. root 702 0.0 0.0 110044 696 tty1 Ss+ Aug19 0:00 /sbin/agetty --noclear tty1 linux
  9. root 703 0.0 0.0 110044 732 hvc0 Ss+ Aug19 0:00 /sbin/agetty --keep-baud 115200 38400 9600 hvc0 vt220
  10. root 3025 0.0 0.0 1520 16 pts/1 Ss+ Aug19 0:00 /bin/sh -c nginx -g "daemon on;" && uwsgi --ini /data/web/uwsgi.ini
  11. root 3694 0.0 0.1 242444 10644 pts/1 Sl+ Aug19 0:01 uwsgi --ini /data/web/uwsgi.ini
  12. root 3992 0.0 0.0 102696 1468 pts/7 Ss+ Aug19 10:49 /usr/local/bin/python /usr/local/bin/gunicorn -w 3 -k gevent -b :5001 manage:app
  13. root 4089 0.0 0.0 11636 8 pts/8 Ss+ Aug19 0:00 /bin/sh -c uwsgi --ini /data/code/uwsgi.ini && nginx -g "daemon off;"

  

可以看出,通过ps命令可以查看当前运行的所有进程,该命令同时列出了PID(进程ID),ps命令可通过指定a和u参数u列出所有进程的详细信息

通过fork函数创建进程

  1. #include<unistd.h>
  2. pid_t fork(void);//成功时返回进程ID,失败时返回-1

  

fork函数将创建调用的进程副本,也就是说,并非根据完全不同的程序创建进程,而是复制正在运行的、调用fork函数的进程。另外,两个进程都将执行fork函数调用后的语句(准确地说是在fork函数返回后)。但因为通过同一个进程、复制相同的内存空间,之后的程序流根据fork函数的返回值加以区分。即利用fork函数的如下特点区分程序执行流程:

  • 父进程:fork函数返回子进程ID
  • 子进程:fork函数返回0

此处,“父进程”指原进程,即调用fork函数的主体,而“子进程”是通过父进程调用fork函数复制出的进程。图1-1展示了调用fork函数后的程序运行流程

图1-1   fork函数的调用

图1-1中可以看到,父进程调用fork函数的同时复制出子进程,并分别得到fork函数的返回值。但复制前,父进程全局变量gval增加到11,将局部变量lval的值增加到25。复制完成后根据fork函数的返回类型区分父子进程,父进程将lval加1,但这不会影响子进程的lval的值。同样,子进程将gval的值加1也不会影响父进程的gval。因为fork函数调用后分成了两个完全不同的进程,只是二者共享同一代码块而已。接下来,我们验证之前所说的内容

fork.c

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. int gval = 10;
  4.  
  5. int main(int argc, char *argv[])
  6. {
  7. pid_t pid;
  8. int lval = 20;
  9. gval++, lval += 5;
  10.  
  11. pid = fork();
  12. if (pid == 0) // if Child Process
  13. gval += 2, lval += 2;
  14. else // if Parent Process
  15. gval -= 2, lval -= 2;
  16.  
  17. if (pid == 0)
  18. printf("Child Proc: [%d, %d] \n", gval, lval);
  19. else
  20. printf("Parent Proc: [%d, %d] \n", gval, lval);
  21. return 0;
  22. }

  

  • 第11行:创建子进程,父进程的pid中存有子进程的ID,子进程的pid是0
  • 第12、18行:子进程执行这两行代码,因为pid为0
  • 第15、20行:父进程执行这两行代码,因为此时pid中存有子进程ID

编译fork.c并运行

  1. # gcc fork.c -o fork
  2. # ./fork
  3. Parent Proc: [9, 23]
  4. Child Proc: [13, 27]

  

从运行结果可以看出,调用fork函数后,父子进程拥有完全独立的内存结构

进程和僵尸进程

文件操作中,关闭文件和打开文件同等重要。同样,进程销毁也和进程创建同等重要。如果未认真对待进程销毁,它们将变成僵尸进程困扰各位。

僵尸进程

进程完成工作后(执行完main函数中的程序后)应被销毁,但有时这些进程变成僵尸进程,占用系统中的重要资源。这种状态下的进程称作“僵尸进程”,这也是给系统带来负担的原因之一。因此,我们应该消灭这种进程

产生僵尸进程的原因

为了防止僵尸进程的产生,先解释产生僵尸进程的原因。利用如下两个示例展示调用fork函数产生子进程的终止方式:

  • 传递参数并调用exit函数
  • main函数中执行return并返回值

向exit函数传递的参数值和main函数的return语句返回的值都会传递给操作系统,而操作系统不会销毁子进程,直到把这些值传递给产生该子进程的父进程,处在这种状态下的进程就是僵尸进程。也就是说,将子进程变成僵尸进程的正是操作系统。既然如此,僵尸进程何时被销毁呢?其实之前已给出答案:当子进程将返回值传递给父进程的时候。那么,如何向父进程传递返回值呢?操作系统不会主动把这些值传递给父进程,只有父进程主动发起请求(函数调用)时,操作系统才会传递该值。换言之,如果父进程未主动要求获得子进程的结束状态值,操作系统将一直保存,并让子进程长时间处于僵尸进程状态。接下来的示例将创建僵尸进程

zombie.c

  1. #include <stdio.h>
  2. #include <unistd.h>
  3.  
  4. int main(int argc, char *argv[])
  5. {
  6. pid_t pid = fork();
  7.  
  8. if (pid == 0) // if Child Process
  9. {
  10. puts("Hi I'am a child process");
  11. }
  12. else
  13. {
  14. printf("Child Process ID: %d \n", pid);
  15. sleep(30); // Sleep 30 sec.
  16. }
  17.  
  18. if (pid == 0)
  19. puts("End child process");
  20. else
  21. puts("End parent process");
  22. return 0;
  23. }

  

  • 第14行:输出子进程ID,可以通过该值查看子进程状态(是否为僵尸进程)
  • 第15行:父进程暂停30秒,如果父进程终止,处于僵尸进程状态的子进程将同时销毁。因此,延缓父进程的执行以验证僵尸进程

编译zombie.c并运行

  1. # ./zombie
  2. Child Process ID: 5507
  3. Hi I'am a child process
  4. End child process
  5. End parent process

  

程序开始运行,在打印出子进程的进程ID后,会停歇30秒,这个时候我们可以趁机看一下5507进程号所对应的进程状态

  1. # ps -ef | grep 5507
  2. root 5507 5506 0 11:44 pts/32 00:00:00 [zombie] <defunct>
  3. root 5509 23062 0 11:45 pts/31 00:00:00 grep --color=auto 5507

  

可以看到,5507对应的进程号的状态为defunct,即为僵尸进程。经过30秒后,随着父进程的终止,子进程也将销毁

销毁僵尸进程1:利用wait函数

如前所述,为了销毁子进程,父进程应主动请求获取子进程的返回值,接下来讨论下发起请求的具体方法,共有两种,其中之一就是调用wait函数

  1. #include <sys/wait.h>
  2. pid_t wait(int *statloc);//成功时返回终止的子进程ID,失败时返回-1

  

调用次函数时如果已有子进程终止,那么子进程终止时传递的返回值(exit函数的参数值、main函数的return返回值)将保存到该函数的参数所指的内存空间。但函数参数指向的单元中还包含其他信息,因此需要通过下列宏进行分离

  • WIFEXITED子进程正常终止时返回真(true)
  • WEXITSTATUS返回子进程的返回值

也就是说,向wait函数传递变量status的地址时,调用wait函数后应编写如下代码

  1. if (WIFEXITED(status))
  2. {
  3. puts("Normal termination!");
  4. printf("Child pass num: %d \n", WEXITSTATUS(status)); //返回值是多少
  5. }

  

根据上述内容编写示例,此示例中不会再让子进程编程僵尸进程

wait.c

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <sys/wait.h>
  5.  
  6. int main(int argc, char *argv[])
  7. {
  8. int status;
  9. pid_t pid = fork();
  10.  
  11. if (pid == 0)
  12. {
  13. return 3;
  14. }
  15. else
  16. {
  17. printf("Child PID: %d \n", pid);
  18. pid = fork();
  19. if (pid == 0)
  20. {
  21. exit(7);
  22. }
  23. else
  24. {
  25. printf("Child PID: %d \n", pid);
  26. wait(&status);
  27. if (WIFEXITED(status))
  28. printf("Child send one: %d \n", WEXITSTATUS(status));
  29.  
  30. wait(&status);
  31. if (WIFEXITED(status))
  32. printf("Child send two: %d \n", WEXITSTATUS(status));
  33. sleep(30); // Sleep 30 sec.
  34. }
  35. }
  36. return 0;
  37. }

  

  • 第9、13行:第9行创建的子进程将在第13行通过main函数中的return语句终止
  • 第18、21行:第18行中创建的子进程将在第21行通过调用exit函数终止
  • 第26行:调用wait函数,之前终止的子进程相关信息将保存到status变量,同时相关子进程被完全销毁
  • 第27、28行:第27行中通过WIFEXITED宏验证子进程是否正常终止,如果正常退出,则调用WEXITSTATUS宏输出子进程的返回值
  • 第30~32行:因为之前创建了两个进程,所以再次调用wait函数和宏
  • 第33行:为暂停父进程终止而插入的代码,此时可以查看子进程状态
  1. # gcc wait.c -o wait
  2. # ./wait
  3. Child PID: 6862
  4. Child PID: 6863
  5. Child send one: 3
  6. Child send two: 7

  

在系统中执行ps命令可以发现,并没有上一个示例中对应PID的进程。这是因为调用了wait函数,完全销毁了子进程,另外两个子进程终止时返回3和7传递给父进程。这就是通过调用wait函数消灭僵尸进程的方法,调用wait函数时,如果没有已终止的子进程,那么程序将阻塞直到有子进程终止,因此需谨慎调用该函数

销毁僵尸进程2:使用waitpid函数

wait函数会引起程序的阻塞,还可以考虑调用waitpid函数,这是防止僵尸进程的第二种方法,也是防止阻塞的方法

  1. #include <sys/wait.h>
  2. pid_t waitpid(pid_t pid, int *statloc, int options);//成功时返回终止的子进程ID(或0),失败时返回-1

  

  • pid:等待终止的目标子进程的ID,若传递-1,则与wait函数相同,可以等待任意子进程终止
  • statloc:与wait函数的statloc具有相同意义
  • options:传递头文件sys/wait.h中声明的常量WNOHANG,即使没有终止的子进程也不会进入阻塞状态,而是返回0并退出函数

下面介绍用上述函数的示例,调用waitpid函数,程序不会阻塞

waitpid.c

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <sys/wait.h>
  4.  
  5. int main(int argc, char *argv[])
  6. {
  7. int status;
  8. pid_t pid = fork();
  9.  
  10. if (pid == 0)
  11. {
  12. sleep(15);
  13. return 24;
  14. }
  15. else
  16. {
  17. while (!waitpid(-1, &status, WNOHANG))
  18. {
  19. sleep(3);
  20. puts("sleep 3sec.");
  21. }
  22.  
  23. if (WIFEXITED(status))
  24. printf("Child send %d \n", WEXITSTATUS(status));
  25. }
  26. return 0;
  27. }

  

  • 第12行:调用sleep函数推迟子进程的执行,这会导致程序延迟15秒
  • 第17行:while循环调用waitpid函数,向第三个参数传递WNOHANG,因此,若之前没有终止的子进程将返回0

编译waitpid.c并运行

  1. # gcc waitpid.c -o waitpid
  2. # ./waitpid
  3. sleep 3sec.
  4. sleep 3sec.
  5. sleep 3sec.
  6. sleep 3sec.
  7. sleep 3sec.
  8. Child send 24

  

可以看出第20行共执行了五次,另外,也证明waitpid函数并未阻塞

TCP/IP网络编程之多进程服务端(一)的更多相关文章

  1. TCP/IP网络编程之多进程服务端(二)

    信号处理 本章接上一章TCP/IP网络编程之多进程服务端(一),在上一章中,我们介绍了进程的创建和销毁,以及如何销毁僵尸进程.前面我们讲过,waitpid是非阻塞等待子进程销毁的函数,但有一个不好的缺 ...

  2. TCP/IP网络编程之多线程服务端的实现(二)

    线程存在的问题和临界区 上一章TCP/IP网络编程之多线程服务端的实现(一)的thread4.c中,我们发现多线程对同一变量进行加减,最后的结果居然不是我们预料之内的.其实,如果多执行几次程序,会发现 ...

  3. TCP/IP网络编程之多线程服务端的实现(一)

    为什么引入线程 为了实现服务端并发处理客户端请求,我们介绍了多进程模型.select和epoll,这三种办法各有优缺点.创建(复制)进程的工作本身会给操作系统带来相当沉重的负担.而且,每个进程有独立的 ...

  4. TCP/IP网络编程之进程间通信

    进程间通信基本概念 进程间通信意味着两个不同进程间可以交换数据,为了完成这一点,操作系统中应提供两个进程可以同时访问的内存空间.但我们知道,进程具有完全独立的内存结构,就连通过fork函数创建的子进程 ...

  5. TCP/IP网络编程之基于TCP的服务端/客户端(二)

    回声客户端问题 上一章TCP/IP网络编程之基于TCP的服务端/客户端(一)中,我们解释了回声客户端所存在的问题,那么单单是客户端的问题,服务端没有任何问题?是的,服务端没有问题,现在先让我们回顾下服 ...

  6. TCP/IP网络编程之基于TCP的服务端/客户端(一)

    理解TCP和UDP 根据数据传输方式的不同,基于网络协议的套接字一般分为TCP套接字和UDP套接字.因为TCP套接字是面向连接的,因此又称为基于流(stream)的套接字.TCP是Transmissi ...

  7. TCP/IP网络编程系列之四(初级)

    TCP/IP网络编程系列之四-基于TCP的服务端/客户端 理解TCP和UDP 根据数据传输方式的不同,基于网络协议的套接字一般分为TCP和UDP套接字.因为TCP套接字是面向连接的,因此又称为基于流的 ...

  8. TCP/IP网络编程系列之三(初级)

    TCP/IP网络编程系列之三-地址族与数据序列 分配给套接字的IP地址和端口 IP是Internet Protocol (网络协议)的简写,是为首发网络数据而分配给计算机的值.端口号并非赋予计算机值, ...

  9. TCP/IP网络编程系列之二(初级)

    套接字类型与协议设置 我们先了解一下创建套接字的那个函数 int socket(int domain,int type,int protocol);成功时返回文件描述符,失败时返回-1.其中,doma ...

随机推荐

  1. jQuery_3_过滤选择器

    过滤选择器(过滤器)类似于CSS3里的伪类,包含 1. 基本过滤器 2. 内容过滤器 3. 可见性过滤器 4. 子元素过滤器 5. 其他方法 一.  基本过滤器 过滤器名 jQuery语法 注释 :f ...

  2. HDU3308 线段树区间合并

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3308 ,简单的线段树区间合并. 线段树的区间合并:一般是要求求最长连续区间,在PushUp()函数中实 ...

  3. js 对象字面量

    对象字面量的输出方式以及定义好处 1.对象字面量的输出方式有两种:传统的'.' 例如:box.name 以及数组方式,只不过用数组方式输出时,方括号里面要用引号括起来 例如:box['name'] v ...

  4. json字符串转换成对象需要注意的问题

    json转换成对象的时候应该尽量避免出现特殊的符号,如“\”这样的字符在转义成数组的时候会被去除掉,最好的例子就是后台返回的内容为存储路径的JSON,这时候最好是把一个斜杠变为两个斜杠,如: [{&q ...

  5. POJ 3666 Making the Grade(区间dp)

    修改序列变成非递减序列,使得目标函数最小.(这题数据有问题,只要求非递减 从左往右考虑,当前a[i]≥前一个数的取值,当固定前一个数的取值的时候我们希望前面操作的花费尽量小. 所以状态可以定义为dp[ ...

  6. Problem A: C语言习题 链表建立,插入,删除,输出

    #include<stdio.h> #include<string.h> #include<stdlib.h> typedef struct student { l ...

  7. 标准输入输出 stdio 流缓冲 buffering in standard streams

    From : http://www.pixelbeat.org/programming/stdio_buffering/ 译者:李秋豪 我发现找出标准流用的是什么缓冲是一件困难的事. 例如下面这个使用 ...

  8. x5webview 微信H5支付

    mWebView.setWebViewClient(new WebViewClient() { // @Override // public boolean shouldOverrideUrlLoad ...

  9. angular6项目中使用echarts图表的方法(有一个坑,引用报错)

    1.安装相关依赖(采用的webpack) npm install ecahrts --save npm install ngx-echarts --save 2.angular.json 配置echa ...

  10. java基础——线程池

    第2章 线程池 2.1 线程池概念 线程池,其实就是一个容纳多个线程的容器,其中的线程可以反复使用,省去了频繁创建线程对象的操作,无需反复创建线程而消耗过多资源. 我们详细的解释一下为什么要使用线程池 ...