0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

你们知道Linux的进程是怎样创建的吗

Linux爱好者 来源:CS指南 作者:大白 2021-11-09 10:46 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

Linux的进程是怎样创建的

Linux系统创建进程都是由已存在的进程创建的(除了0号进程),被创建的进程叫做子进程,创建子进程的进程就做父进程。这句话是不是有点熟悉,没错,Linux进程串起来也是一颗树的结构。就像下面这样:

01d5842e-3f30-11ec-9195-dac502259ad0.png

在Linux中,为了创建一个子进程,父进程用系统调用fork来创建子进程。fork()其实就是把父进程复制了一份(子进程有自己的特性,比如标识、状态、数据空间等;子进程和父进程共同使用程序代码、共用时间片等)。

可以看下面这段代码:

#include
#include

int main()
{
int p_num = 0;
int c_num = 0;
int pid = fork();
if(pid == 0) //返回的pid为0为子进程
{
c_num++;
}
else
{
p_num++; //返回的pid大于0为父进程
}
printf("p_num=%d, c_num=%d
",p_num,c_num);
printf("pid=%d
",pid);
return 0;
}
//运行结果如下所示
p_num=1, c_num=0
pid=36101
p_num=0, c_num=1
pid=0

大家看,代码中调用了fork以后,之后的程序被执行了两遍。子进程和父进程各自的变量互相没有受到干扰。不过子进程和父进程执行的是相同的代码,子进程和父进程资源占用情况如下图所示:

02068b78-3f30-11ec-9195-dac502259ad0.png

大家可以看出,通过fork后,子进程并没有和父进程独立开,用的是相同的代码。另外还有一个问题时,这个时候子进程的时间片是和父进程一分为二来共享的。这样我创建子进程还有什么意义?为了彻底将父进程和子进程分离开来,就要用到一个系统调用 execv()。

看下面这段代码:

//process.c
#include
#include

int main()
{
int pid = fork();
if(pid == 0)
{
execv("./test.o",NULL);  //test.o是一个经过编译的c语言文件,这里记得要放test.o的绝对路径
}
printf("This is parent process
");
return 0;
}

//test.c
#include
int main()
{
printf("This is child process");
return 0;
}

//运行结果如下所示
This is parent process
This is child process

通过上面的代码可以看出,从系统调用 execv() 后,子进程直接走自己的代码了,没有像前一段代码一样把后面的代码执行了两次。通过调用 execv(),子进程和父进程就基本分离开了。

结合系统继续看Linux的进程树是什么样的

好了,通过上面的介绍,大家应该对进程是怎么创建的有一定的了解。想继续学习的我们来接着上强度。

我们在 Linux 系统上通过 ps - ef 命令查看系统目前的进程:

/[root@localhost lucas]# ps -ef
UID          PID    PPID  C STIME TTY          TIME CMD
root           1       0  3 21:41 ?        00:02:38 /usr/lib/systemd/systemd --s
root           2       0  0 21:41 ?        00:00:07 [kthreadd]
root           3       2  0 21:41 ?        00:00:00 [rcu_gp]
root           4       2  0 21:41 ?        00:00:00 [rcu_par_gp]
...
rtkit       1151       1  0 21:41 ?        00:00:14 /usr/libexec/rtkit-daemon
root        1152       1  0 21:41 ?        00:00:00 /usr/sbin/ModemManager
avahi       1155       1  0 21:41 ?        00:00:06 avahi-daemon: running [linux
root        1159       1  0 21:41 ?        00:00:02 /usr/lib/systemd/systemd-mac

我来解释上表是什么意思。

首先,每一个进程都要所属一个用户,UID 就是用户的标识符(通过 root 用户创建的进程 UID 就是 root,如果我自己创建的话就应该是我的用户名,比如我的名字 "dabai")。

其次每一个进程都要有一个 ID 来表示这个进程,PID 就表示的是当前进程的 id。

最后,上文提到除了 0 号进程,每一个进程都是由他的父进程创建的,PPID 就表示当前进程的父进程 id。

通过 0 号进程创建 1 号进程和 2 号进程,然后通过 1 号进程去创建用户态进程,再通过 2 号进程创建内核态进程,就生成了 Linux 进程树。

0248cfec-3f30-11ec-9195-dac502259ad0.png

「什么是0号进程、1号进程以及2号进程?」

0号进程:在内核初始化的过程中,会先通过指令 struct task_struct init_task = INIT_TASK(init_task) 创建 0 号进程。这是唯一一个没有通过 fork 或者 kernel_thread 产生的进程。是进程列表的第一个。但是这个进程不是实际意义上的进程,类似与链表头。所以虽然 0 号进程是在内核态创建的,但不能说 0 号进程是内核态的第一个进程,反而要说 2 号进程是内核态的第一个进程。

1号进程:通过调用指令 kernel_thread(kernel_init, NULL, CLONE_FS) 从内核态切换到用户态来创建的,1号进程是所有用户态的祖先。

2号进程:通过调用指令 kernel_thread(kthreadd, NULL, ClONE_FS | CLONE_FILES) 来创建,2号进程负责所有内核态的进程的调度和管理,是内核态所有进程的祖先。(注意,内核态不区分线程和进程,所以说进程和线程都可以,都是任务)

「为什么要先创建 0 号进程,而不直接创建 1 号进程?」

现在对于为什么要先创建 0 号进程而不直接创建1号和2号进程有许多讨论。我认为...算了,我不认为了,一展开讲这篇文章又收不了尾了,以后可以专门写一篇文章来论述这里。简单来说就是Linux 的第一个进程不适合是一个真进程,需要一个没有数据之类东西的假进程。

「为什么要区分用户态和内核态?」

因为有了多个进程,对于关键资源来说,就会产生争用以及误操作破坏资源等情况。这时就需要对资源的访问权限进行一定的限制。x86 提供了分层的权限机制,内核态具有最高的访问权限,而用户态访问核心资源时必须要切换到内核态才可以访问。

好了,我看了下字数,这篇文章已经不少了,接下来我还会继续去分享进程和线程的更多细节,也会根据读者的反馈在已完成的文章上不断完善,欢迎大家持续关注呀!

参考资料:

【1】Linux进程的创建与管理:https://blog.csdn.net/qq_38410730/article/details/81193118

【2】极客时间:《趣谈Linux操作系统

编辑:jq
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Linux
    +关注

    关注

    88

    文章

    11817

    浏览量

    219555
  • PID
    PID
    +关注

    关注

    38

    文章

    1505

    浏览量

    91905
  • 代码
    +关注

    关注

    30

    文章

    4976

    浏览量

    74384

原文标题:Linux 的进程是怎样创建的

文章出处:【微信号:LinuxHub,微信公众号:Linux爱好者】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    飞凌嵌入式ElfBoard-进程进程状态

    Linux系统中,进程状态对于系统调度、资源分配和管理非常重要,因为它表示了进程当前的执行状况和资源使用情况。在Linux 系统中使用ps -aux 命令可观察到
    发表于 03-27 09:12

    飞凌嵌入式ElfBoard-进程的相关信息之用户ID和组ID

    Linux操作系统中,用户ID(UID) 和 组ID(GID)对进程有着重要的作用,主要影响到进程的权限管理、资源访问以及操作系统的安全性。进程的用户ID和组ID代表了该
    发表于 03-12 17:13

    飞凌嵌入式ElfBoard-进程的相关信息之父进程和子进程

    进程创建时,创建进程是新进程的父进程,新进程
    发表于 03-12 17:12

    Linux内核三大核心模块深度解析:调度、内存与I/O

    优与系统开发的基石。 一、进程调度:从CFS到EEVDF的演进 进程调度负责决定哪个任务获得CPU执行权。在Linux中,进程是资源的容器,而线程才是实际的调度单元,两者均通过kern
    的头像 发表于 03-12 09:00 234次阅读
    <b class='flag-5'>Linux</b>内核三大核心模块深度解析:调度、内存与I/O

    飞凌嵌入式ElfBoard-进程之什么是进程

    Linux系统中,有些基本命令能够查看到进程的信息。例如ps、top、pgrep、pstree等;这些命令为用户提供了查看和管理Linux进程信息的多种功能。通过合理使用这些命令,用
    发表于 03-02 08:49

    Linux进程管理不用愁!这6个工具帮你搞定90%场景

    Linux 系统中,进程是资源分配的基本单位,无论是服务器运维、程序调试还是日常使用,掌握进程管理工具都是必备技能。今天就带大家梳理 6 个最常用的进程管理工具,从查看
    的头像 发表于 02-04 16:23 2875次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>进程</b>管理不用愁!这6个工具帮你搞定90%场景

    Linux进程树分析工具pstree详解与实战指南(另一视角优化Linux系统)

    Linux 系统开发与运维中,理解进程的运行状态和相互关系是排查问题、优化性能的基础。pstree 作为一款轻量高效的进程树可视化工具,能直观展示系统中所有进程的父子关系,为系统分
    的头像 发表于 02-04 16:21 903次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>进程</b>树分析工具pstree详解与实战指南(另一视角优化<b class='flag-5'>Linux</b>系统)

    飞凌嵌入式ElfBoard-进程之什么是进程

    的开始和结束在 Linux 操作系统中,程序是静态的可执行文件,而进程是动态的实体。从程序变为进程的过程涉及多个步骤,其中还包括系统资源的管理和初始化。1.当用户在 Shell 中输入命令时,Shell
    发表于 01-26 08:42

    进程概念和特征

    进程的基本情况和运行状态,进而控制和管理进程。相应地,由程序段、相关数据段和PCB三部分构成了进程映像(进程实体)。所谓创建
    发表于 01-15 06:39

    进程的控制

    进程控制的主要功能是对系统中的所有进程实施有效的管理,它具有创建进程、撤销已有进程、实现进程
    发表于 01-15 06:05

    深入Linux内核:进程调度的核心逻辑与实现细节

    Linux系统中,进程调度就像一位精明的“CPU管理员”——它决定着哪个进程能优先使用CPU,多久切换一次进程,如何平衡系统响应速度与资源利用率。小到桌面应用的流畅点击,大到服务器的
    的头像 发表于 12-24 07:05 4584次阅读
    深入<b class='flag-5'>Linux</b>内核:<b class='flag-5'>进程</b>调度的核心逻辑与实现细节

    解析Linux进程、线程和协程

    系统的稳定性和安全性。 (3)创建与销毁:Linux使用fork()系统调用来创建进程,通过exit()来终止进程。 线程(Thread
    发表于 12-22 11:00

    Linux进程间通信(IPC)全解析:从管道到 Socket,一篇讲透

    在 Linux 世界里,进程并非孤立存在。无论是后台服务协作(如 Web 服务器与数据库)、命令行工具联动(如ps | grep),还是复杂应用的模块通信,都离不开 进程间通信(IPC
    的头像 发表于 11-14 21:38 1.3w次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>进程</b>间通信(IPC)全解析:从管道到 Socket,一篇讲透

    【HZ-T536开发板免费体验】—— linux创建线程

    任务,并行是多个处理器或多核处理器同时执行不同的任务。 Linux系统中进行多线程编程时,会涉及到主线程和子线程的操作: 1)主线程是程序的执行入口,它是程序中第一个创建的线程。 2)子线程,是主线
    发表于 09-01 21:31

    【HZ-T536开发板免费体验】—— linux 进程创建

    Linux进程通信方式有这几种: 1。管道 2。信号量 3。消息队列 4。共享内存 在本帖子中,我会讲解fork(),exit()系统调用的实践。通过应用编程来实现系统调用。 1,进程创建
    发表于 09-01 20:49