【TCP/IP】多进程服务器的实现(进阶) - 进程的概念及fork函数

文章介绍了进程的概念,作为操作系统资源分配的基本单位,以及进程ID的分配和查看方法。重点讲解了fork函数在创建进程中的作用,如何通过fork函数创建子进程,并通过示例代码解释了父进程和子进程如何独立执行并改变各自的状态。
摘要由CSDN通过智能技术生成

目录

进程的概念及应用

进程的定义

进程的ID

fork函数(进程创建函数)


        多进程(以及多线程)是现代计算机网络的精髓。在之前,我们所做的诸如回声服务器、回声客户端、文件收发等都是偏向基础的单进程应用。而经过前面的铺垫,我们对Socket也有了一定了解。接下来,让我们一起正式开始探索真正的网络编程吧!

进程的概念及应用

进程的定义

        进程(Process)是计算机中程序关于某数据集合上的一次运行活动,是系统进行资源分配的基本单位,是操作系统结构的基础。在早期面向进程设计的计算机结构中,进程是程序的基本执行实体;在当代面向线程设计的计算机结构中,进程是线程的容器。程序是指令、数据及其组织形式的描述,进程是程序的实体。

补充:

宏观上,当有多个进程运行在操作系统上时,我们看到的效果是这些程序可以同时得到响应。而从微观上看,这些(程序的)进程并非同时被执行,而是通过在CPU分配的时间片中,借助CPU的高效计算能力,被反复快速地交替执行,进而让我们视觉上看到的是“同时进行”的效果。

进程的ID

        在操作系统中,所有进程创建时都会被分配一个ID,我们将这个ID称为“进程ID”,其值为大于2的整数。值为1的进程ID通常被分配给用于协助操作系统的系统应用程序。

        在Linux中,我们可以通过 ps 命令来查看当前系统正在运行的进程,其用法如下:

                        ps [选项,见下表] 

基本选项含义
 -A, -e所有进程
 -a所有tty终端下的进程,除了对话期首进程
a所有tty终端下的进程,包括其他用户
-d全部进程,除了会话期首进程
-N, --deselect反选进程(显示进程中未显示的部分)
r运行中的进程
T当前终端下的所有进程
x非tty终端控制下的进程

        如图所示:

        根据需要,还可进一步显示进程细节,具体用法可以通过 ps --help all 查阅,在这里不再赘述。

fork函数(进程创建函数)

        在多进程服务器端中,我们常用fork函数来创建进程,具体用法如下:

#include <unistd.h>

pid_t fork(void);

//成功时返回进程ID ,失败时返回-1

        fork函数会创建调用该函数的进程的副本,即复制一份当前调用了fork函数的进程。副本进程(子进程)创建完成后,会同原进程(父进程)一起执行fork函数调用完成后的语句。在判断原进程和副本进程时,需要通过fork函数的返回值来区分。

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

        父进程(Parent Process)指的是调用fork函数的主体,而子进程(Child Process)是通过父进程调用fork函数复制得来的进程。 

        调用fork函数后的程序运行流程如下图所示:

        从图中可以看出,父进程调用fork函数时创建了一份子进程,并由pid_t变量接收fork函数返回的进程ID值。在这之前,子进程继承了父进程之前的变量值(gval、lval)。不过在创建完成后,父进程和子进程的ID并不同,在父进程中将执行if条件满足pid!=0的语句块,即lval++;而子进程将执行pid==0的语句块,即gval++。

        让我们来验证下思路:

        fork.cpp

#include <stdio.h>
#include <unistd.h>

int gval = 10;

int main(int argc, char *argv[])
{
    pid_t pid;
    int lval = 20;
    gval++;    // gval -> 11
    lval += 5; // lval -> 25

    pid = fork();
    if (pid == 0) // 子进程
    {
        gval += 1; // gval -> 12
        lval += 2; // lval -> 27
    }
    else // 父进程
    {
        gval -= 1; // gval -> 10
        lval -= 2; // lval -> 23
    }

    pid == 0 ? printf("Child Process: gval = %d, lval = %d \n", gval, lval)
             : printf("Parent Process: gval = %d, lval = %d \n", gval, lval);
    return 0;
}

        输出结果:

        从运行结果来看,验证了我们之前的想法的正确性。需要注意的是,调用fork函数后,父进程之前的变量及其值在子进程中是一致的(也拷贝了一份),但父、子进程又有彼此完全独立的内存结构,因此在fork函数操作完成后执行的下阶段函数互不影响,结果彼此独立。 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

干吃咖啡豆

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值