0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

你們知道Linux的進(jìn)程是怎樣創(chuàng)建的嗎

Linux愛好者 ? 來源:CS指南 ? 作者:大白 ? 2021-11-09 10:46 ? 次閱讀

Linux的進(jìn)程是怎樣創(chuàng)建的

Linux系統(tǒng)創(chuàng)建進(jìn)程都是由已存在的進(jìn)程創(chuàng)建的(除了0號進(jìn)程),被創(chuàng)建的進(jìn)程叫做子進(jìn)程,創(chuàng)建子進(jìn)程的進(jìn)程就做父進(jìn)程。這句話是不是有點(diǎn)熟悉,沒錯(cuò),Linux進(jìn)程串起來也是一顆樹的結(jié)構(gòu)。就像下面這樣:

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

在Linux中,為了創(chuàng)建一個(gè)子進(jìn)程,父進(jìn)程用系統(tǒng)調(diào)用fork來創(chuàng)建子進(jìn)程。fork()其實(shí)就是把父進(jìn)程復(fù)制了一份(子進(jìn)程有自己的特性,比如標(biāo)識、狀態(tài)、數(shù)據(jù)空間等;子進(jìn)程和父進(jìn)程共同使用程序代碼、共用時(shí)間片等)。

可以看下面這段代碼:

#include
#include

int main()
{
int p_num = 0;
int c_num = 0;
int pid = fork();
if(pid == 0) //返回的pid為0為子進(jìn)程
{
c_num++;
}
else
{
p_num++; //返回的pid大于0為父進(jìn)程
}
printf("p_num=%d, c_num=%d
",p_num,c_num);
printf("pid=%d
",pid);
return 0;
}
//運(yùn)行結(jié)果如下所示
p_num=1, c_num=0
pid=36101
p_num=0, c_num=1
pid=0

大家看,代碼中調(diào)用了fork以后,之后的程序被執(zhí)行了兩遍。子進(jìn)程和父進(jìn)程各自的變量互相沒有受到干擾。不過子進(jìn)程和父進(jìn)程執(zhí)行的是相同的代碼,子進(jìn)程和父進(jìn)程資源占用情況如下圖所示:

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

大家可以看出,通過fork后,子進(jìn)程并沒有和父進(jìn)程獨(dú)立開,用的是相同的代碼。另外還有一個(gè)問題時(shí),這個(gè)時(shí)候子進(jìn)程的時(shí)間片是和父進(jìn)程一分為二來共享的。這樣我創(chuàng)建子進(jìn)程還有什么意義?為了徹底將父進(jìn)程和子進(jìn)程分離開來,就要用到一個(gè)系統(tǒng)調(diào)用 execv()。

看下面這段代碼:

//process.c
#include
#include

int main()
{
int pid = fork();
if(pid == 0)
{
execv("./test.o",NULL);  //test.o是一個(gè)經(jīng)過編譯的c語言文件,這里記得要放test.o的絕對路徑
}
printf("This is parent process
");
return 0;
}

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

//運(yùn)行結(jié)果如下所示
This is parent process
This is child process

通過上面的代碼可以看出,從系統(tǒng)調(diào)用 execv() 后,子進(jìn)程直接走自己的代碼了,沒有像前一段代碼一樣把后面的代碼執(zhí)行了兩次。通過調(diào)用 execv(),子進(jìn)程和父進(jìn)程就基本分離開了。

結(jié)合系統(tǒng)繼續(xù)看Linux的進(jìn)程樹是什么樣的

好了,通過上面的介紹,大家應(yīng)該對進(jìn)程是怎么創(chuàng)建的有一定的了解。想繼續(xù)學(xué)習(xí)的我們來接著上強(qiáng)度。

我們在 Linux 系統(tǒng)上通過 ps - ef 命令查看系統(tǒng)目前的進(jìn)程:

/[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

我來解釋上表是什么意思。

首先,每一個(gè)進(jìn)程都要所屬一個(gè)用戶,UID 就是用戶的標(biāo)識符(通過 root 用戶創(chuàng)建的進(jìn)程 UID 就是 root,如果我自己創(chuàng)建的話就應(yīng)該是我的用戶名,比如我的名字 "dabai")。

其次每一個(gè)進(jìn)程都要有一個(gè) ID 來表示這個(gè)進(jìn)程,PID 就表示的是當(dāng)前進(jìn)程的 id。

最后,上文提到除了 0 號進(jìn)程,每一個(gè)進(jìn)程都是由他的父進(jìn)程創(chuàng)建的,PPID 就表示當(dāng)前進(jìn)程的父進(jìn)程 id。

通過 0 號進(jìn)程創(chuàng)建 1 號進(jìn)程和 2 號進(jìn)程,然后通過 1 號進(jìn)程去創(chuàng)建用戶態(tài)進(jìn)程,再通過 2 號進(jìn)程創(chuàng)建內(nèi)核態(tài)進(jìn)程,就生成了 Linux 進(jìn)程樹。

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

「什么是0號進(jìn)程、1號進(jìn)程以及2號進(jìn)程?」。

0號進(jìn)程:在內(nèi)核初始化的過程中,會先通過指令 struct task_struct init_task = INIT_TASK(init_task) 創(chuàng)建 0 號進(jìn)程。這是唯一一個(gè)沒有通過 fork 或者 kernel_thread 產(chǎn)生的進(jìn)程。是進(jìn)程列表的第一個(gè)。但是這個(gè)進(jìn)程不是實(shí)際意義上的進(jìn)程,類似與鏈表頭。所以雖然 0 號進(jìn)程是在內(nèi)核態(tài)創(chuàng)建的,但不能說 0 號進(jìn)程是內(nèi)核態(tài)的第一個(gè)進(jìn)程,反而要說 2 號進(jìn)程是內(nèi)核態(tài)的第一個(gè)進(jìn)程。

1號進(jìn)程:通過調(diào)用指令 kernel_thread(kernel_init, NULL, CLONE_FS) 從內(nèi)核態(tài)切換到用戶態(tài)來創(chuàng)建的,1號進(jìn)程是所有用戶態(tài)的祖先。

2號進(jìn)程:通過調(diào)用指令 kernel_thread(kthreadd, NULL, ClONE_FS | CLONE_FILES) 來創(chuàng)建,2號進(jìn)程負(fù)責(zé)所有內(nèi)核態(tài)的進(jìn)程的調(diào)度和管理,是內(nèi)核態(tài)所有進(jìn)程的祖先。(注意,內(nèi)核態(tài)不區(qū)分線程和進(jìn)程,所以說進(jìn)程和線程都可以,都是任務(wù))

「為什么要先創(chuàng)建 0 號進(jìn)程,而不直接創(chuàng)建 1 號進(jìn)程?」

現(xiàn)在對于為什么要先創(chuàng)建 0 號進(jìn)程而不直接創(chuàng)建1號和2號進(jìn)程有許多討論。我認(rèn)為...算了,我不認(rèn)為了,一展開講這篇文章又收不了尾了,以后可以專門寫一篇文章來論述這里。簡單來說就是Linux 的第一個(gè)進(jìn)程不適合是一個(gè)真進(jìn)程,需要一個(gè)沒有數(shù)據(jù)之類東西的假進(jìn)程。

「為什么要區(qū)分用戶態(tài)和內(nèi)核態(tài)?」

因?yàn)橛辛硕鄠€(gè)進(jìn)程,對于關(guān)鍵資源來說,就會產(chǎn)生爭用以及誤操作破壞資源等情況。這時(shí)就需要對資源的訪問權(quán)限進(jìn)行一定的限制。x86 提供了分層的權(quán)限機(jī)制,內(nèi)核態(tài)具有最高的訪問權(quán)限,而用戶態(tài)訪問核心資源時(shí)必須要切換到內(nèi)核態(tài)才可以訪問。

好了,我看了下字?jǐn)?shù),這篇文章已經(jīng)不少了,接下來我還會繼續(xù)去分享進(jìn)程和線程的更多細(xì)節(jié),也會根據(jù)讀者的反饋在已完成的文章上不斷完善,歡迎大家持續(xù)關(guān)注呀!

參考資料

【1】Linux進(jìn)程的創(chuàng)建與管理:https://blog.csdn.net/qq_38410730/article/details/81193118

【2】極客時(shí)間:《趣談Linux操作系統(tǒng)

編輯:jq
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11207

    瀏覽量

    208717
  • PID
    PID
    +關(guān)注

    關(guān)注

    35

    文章

    1469

    瀏覽量

    85193
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4722

    瀏覽量

    68231

原文標(biāo)題:Linux 的進(jìn)程是怎樣創(chuàng)建的

文章出處:【微信號:LinuxHub,微信公眾號:Linux愛好者】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    一文搞懂Linux進(jìn)程的睡眠和喚醒

    操作系統(tǒng)進(jìn)行管理; 父進(jìn)程號:(PPID:Parent Process ID):用于描述一個(gè)進(jìn)程的直接父進(jìn)程的標(biāo)識符,每個(gè)進(jìn)程創(chuàng)建時(shí)都會
    發(fā)表于 11-04 15:15

    linux常用性能優(yōu)化方法

    占用一個(gè)本地端口號(與 TCP 協(xié)議端口號不一樣),相當(dāng)于一個(gè)進(jìn)程,便于與其它進(jìn)程進(jìn)行交互。而Linux內(nèi)核的TCP/IP 協(xié)議實(shí)現(xiàn)模塊對本地端口號的范圍進(jìn)行了限制。當(dāng)端口號用盡,就會出現(xiàn)這種錯(cuò)誤了。
    的頭像 發(fā)表于 10-23 13:51 ?97次閱讀

    Linux用戶身份與進(jìn)程權(quán)限詳解

    在學(xué)習(xí) Linux 系統(tǒng)權(quán)限相關(guān)的主題時(shí),我們首先關(guān)注的基本都是文件的 ugo 權(quán)限。ugo 權(quán)限信息是文件的屬性,它指明了用戶與文件之間的關(guān)系。但是真正操作文件的卻是進(jìn)程,也就是說用戶所擁有的文件
    的頭像 發(fā)表于 10-23 11:41 ?201次閱讀
    <b class='flag-5'>Linux</b>用戶身份與<b class='flag-5'>進(jìn)程</b>權(quán)限詳解

    深入探討Linux進(jìn)程調(diào)度器

    Linux操作系統(tǒng)作為一個(gè)開源且廣泛應(yīng)用的操作系統(tǒng),其內(nèi)核設(shè)計(jì)包含了許多核心功能,而進(jìn)程調(diào)度器(Scheduler)就是其中一個(gè)至關(guān)重要的模塊。進(jìn)程調(diào)度器負(fù)責(zé)決定在任何給定的時(shí)刻哪個(gè)進(jìn)程
    的頭像 發(fā)表于 08-13 13:36 ?876次閱讀
    深入探討<b class='flag-5'>Linux</b>的<b class='flag-5'>進(jìn)程</b>調(diào)度器

    Linux添加磁盤創(chuàng)建分區(qū)、掛載

    Linux添加磁盤創(chuàng)建分區(qū)、掛載
    發(fā)表于 04-20 17:49 ?442次閱讀
    <b class='flag-5'>Linux</b>添加磁盤<b class='flag-5'>創(chuàng)建</b>分區(qū)、掛載

    linux下查詢進(jìn)程占用的內(nèi)存方法有哪些?

    linux下查詢進(jìn)程占用的內(nèi)存方法
    發(fā)表于 04-08 06:03

    聊聊怎樣挑選合適的二極管型號

    各位朋友,你們知道嗎,那些微不足道的小元件,比如二極管,是怎樣在電路中扮演著關(guān)鍵角色的?二極管,它可是電子世界中的“流量守門員”,確保電流只能朝著一個(gè)方向流動(dòng)。
    的頭像 發(fā)表于 03-26 15:35 ?759次閱讀
    聊聊<b class='flag-5'>怎樣</b>挑選合適的二極管型號

    淺談Linux進(jìn)程

    進(jìn)程和程序的區(qū)別: 進(jìn)程是動(dòng)態(tài)的,程序是靜態(tài)的 一、進(jìn)程創(chuàng)建(fork()函數(shù)) int main(){ pid_t pid; pid=fork(); if(pid?????>0
    的頭像 發(fā)表于 01-28 15:54 ?234次閱讀
    淺談<b class='flag-5'>Linux</b>的<b class='flag-5'>進(jìn)程</b>

    linux內(nèi)核主要由哪幾個(gè)部分組成,作用是什么

    Linux內(nèi)核主要由以下幾個(gè)部分組成: 進(jìn)程管理:Linux內(nèi)核負(fù)責(zé)管理和調(diào)度系統(tǒng)中的進(jìn)程。它通過進(jìn)程調(diào)度算法來決定哪個(gè)
    的頭像 發(fā)表于 01-22 14:34 ?2565次閱讀

    【從0開始創(chuàng)建AWTK應(yīng)用程序】編譯應(yīng)用到嵌入式Linux平臺運(yùn)行

    AWTK是基于C語言開發(fā)的跨平臺GUI框架。本系列文章介紹如何從0開始創(chuàng)建AWTK應(yīng)用程序,包括搭建開發(fā)調(diào)試環(huán)境、使用AWTK創(chuàng)建Hello工程并在模擬器上運(yùn)行、將AWTK應(yīng)用程序移植到其它平臺
    的頭像 發(fā)表于 12-07 12:08 ?649次閱讀
    【從0開始<b class='flag-5'>創(chuàng)建</b>AWTK應(yīng)用程序】編譯應(yīng)用到嵌入式<b class='flag-5'>Linux</b>平臺運(yùn)行

    Linux進(jìn)程、線程和協(xié)程的基礎(chǔ)概念

    進(jìn)程是計(jì)算機(jī)中運(yùn)行的程序的實(shí)例,它是操作系統(tǒng)中最基本的執(zhí)行單元之一。每個(gè)進(jìn)程都有自己的獨(dú)立內(nèi)存空間、系統(tǒng)資源和代碼執(zhí)行流。這意味著一個(gè)進(jìn)程的崩潰通常不會影響其他進(jìn)程,
    的頭像 發(fā)表于 12-06 09:22 ?748次閱讀

    linux查看weblogic進(jìn)程

    Linux操作系統(tǒng)中,WebLogic是一種常用的Java應(yīng)用服務(wù)器,用于部署和管理企業(yè)級Java應(yīng)用程序。為了確保WebLogic服務(wù)器正常運(yùn)行,有時(shí)我們需要查看WebLogic進(jìn)程以了解其狀態(tài)
    的頭像 發(fā)表于 12-05 16:07 ?1745次閱讀

    Linux進(jìn)程通信的方法

    進(jìn)程是操作系統(tǒng)的概念,每當(dāng)我們執(zhí)行一個(gè)程序時(shí),對于操作系統(tǒng)來講就創(chuàng)建了一個(gè)進(jìn)程,在這個(gè)過程中,伴隨著資源的分配和釋放??梢哉J(rèn)為進(jìn)程是一個(gè)程序的一次執(zhí)行過程。
    的頭像 發(fā)表于 11-29 14:45 ?683次閱讀
    <b class='flag-5'>Linux</b>下<b class='flag-5'>進(jìn)程</b>通信的方法

    進(jìn)程響應(yīng)時(shí)間是指什么

    進(jìn)程響應(yīng)時(shí)間是指從發(fā)出請求到收到響應(yīng)的時(shí)間間隔,是衡量系統(tǒng)性能和用戶體驗(yàn)的重要指標(biāo)之一。在計(jì)算機(jī)系統(tǒng)中,進(jìn)程是指一個(gè)正在運(yùn)行的程序?qū)嵗?。?dāng)用戶發(fā)出請求,系統(tǒng)會創(chuàng)建一個(gè)新的進(jìn)程來處理該請
    的頭像 發(fā)表于 11-17 11:31 ?929次閱讀

    總結(jié)linux命令行的主要用法

    和目錄操作、權(quán)限管理、進(jìn)程管理等內(nèi)容。 1.常用命令 1.1 文件和目錄操作 文件和目錄是Linux系統(tǒng)中最基本的單位,對它們的操作包括創(chuàng)建、刪除、復(fù)制、移動(dòng)、重命名等。常用的命令包括: cd:切換當(dāng)前工作目錄。 pwd:顯示當(dāng)
    的頭像 發(fā)表于 11-17 10:19 ?599次閱讀