成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

面試官,您確定要讓我展開講進程嗎?

系統 Linux
Linux系統創建進程都是由已存在的進程創建的(除了0號進程),被創建的進程叫做子進程,創建子進程的進程就做父進程。這句話是不是有點熟悉,沒錯,Linux進程串起來也是一顆樹的結構。

[[426358]]

本文轉載自微信公眾號「CS指南」,作者大白。轉載本文請聯系CS指南公眾號。

大家好呀,我是大白!準備了好久,我的 Linux 系列文章終于和大家見面了。最近為了籌備 Linux 的文章,以程序員的視角看城市系列都沒時間寫了,最近不光讀者催我,一些喜歡我城市系列文章的號主也在催我,哈哈。現在我的第一篇寫 Linux 的文章終于寫出來了,第二篇、第三篇會越來越快,質量也會越來越高,希望大家多多支持我呀!

面試中經常會被問到進程和線程的區別,你給面試官回答一個進程里包含許多線程他又不太高興,嫌你答的太少了,要掛你。這個系列我們就以 Linux 中的進程和線程為例,好好的拆開講一講里面的細節。今天這篇文章先展開講一講 Linux 的進程是如何創建的。這個系列還會不斷更新,歡迎持續關注呀!

Linux的進程是怎樣創建的

Linux系統創建進程都是由已存在的進程創建的(除了0號進程),被創建的進程叫做子進程,創建子進程的進程就做父進程。這句話是不是有點熟悉,沒錯,Linux進程串起來也是一顆樹的結構。就像下面這樣:

在Linux中,為了創建一個子進程,父進程用系統調用fork來創建子進程。fork()其實就是把父進程復制了一份(子進程有自己的特性,比如標識、狀態、數據空間等;子進程和父進程共同使用程序代碼、共用時間片等)。

可以看下面這段代碼:

  1. #include<stdio.h> 
  2. #include<unistd.h> 
  3.  
  4. int main() 
  5.     int p_num = 0; 
  6.     int c_num = 0; 
  7.     int pid = fork(); 
  8.     if(pid == 0) //返回的pid為0為子進程 
  9.     { 
  10.         c_num++; 
  11.     } 
  12.     else 
  13.     { 
  14.         p_num++; //返回的pid大于0為父進程 
  15.     } 
  16.     printf("p_num=%d, c_num=%d\n",p_num,c_num); 
  17.     printf("pid=%d\n",pid); 
  18.     return 0; 
  19. //運行結果如下所示 
  20. p_num=1, c_num=0 
  21. pid=36101 
  22. p_num=0, c_num=1 
  23. pid=0 

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

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

看下面這段代碼:

  1. //process.c 
  2. #include<stdio.h> 
  3. #include<unistd.h> 
  4.  
  5. int main() 
  6.     int pid = fork(); 
  7.     if(pid == 0) 
  8.     { 
  9.         execv("./test.o",NULL);  //test.o是一個經過編譯的c語言文件,這里記得要放test.o的絕對路徑 
  10.     } 
  11.     printf("This is parent process\n"); 
  12.     return 0; 
  13.  
  14. //test.c 
  15. #include<stdio.h> 
  16. int main() 
  17.     printf("This is child process"); 
  18.     return 0; 
  19.  
  20. //運行結果如下所示 
  21. This is parent process 
  22. This is child process 

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

結合系統繼續看Linux的進程樹是什么樣的

好了,通過上面的介紹,大家應該對進程是怎么創建的有一定的了解。想繼續學習的我們來接著上強度。

我們在 Linux 系統上通過 ps - ef 命令查看系統目前的進程:

  1. /[root@localhost lucas]# ps -ef 
  2. UID          PID    PPID  C STIME TTY          TIME CMD 
  3. root           1       0  3 21:41 ?        00:02:38 /usr/lib/systemd/systemd --s 
  4. root           2       0  0 21:41 ?        00:00:07 [kthreadd] 
  5. root           3       2  0 21:41 ?        00:00:00 [rcu_gp] 
  6. root           4       2  0 21:41 ?        00:00:00 [rcu_par_gp] 
  7. ... 
  8. rtkit       1151       1  0 21:41 ?        00:00:14 /usr/libexec/rtkit-daemon 
  9. root        1152       1  0 21:41 ?        00:00:00 /usr/sbin/ModemManager 
  10. avahi       1155       1  0 21:41 ?        00:00:06 avahi-daemon: running [linux 
  11. 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 進程樹。

「什么是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操作系統》

 

責任編輯:武曉燕 來源: CS指南
相關推薦

2022-07-18 13:59:43

Redis單線程進程

2021-12-02 08:19:06

MVCC面試數據庫

2023-02-16 07:30:38

引用計數算法

2024-09-03 07:58:46

2020-03-10 08:01:05

Java堆內存線程共享

2020-07-02 07:52:11

RedisHash映射

2020-11-10 13:47:29

String源碼長度限制

2022-07-26 08:40:42

Java并發工具類

2022-08-02 06:31:32

Java并發工具類

2020-12-23 13:29:15

微服務架構面試官

2020-02-25 16:56:02

面試官有話想說

2022-05-23 08:43:02

BigIntJavaScript內置對象

2021-12-13 09:10:48

equalshashCodeJava

2022-06-30 08:14:05

Java阻塞隊列

2022-07-11 10:47:46

容器JAVA

2021-12-03 06:59:23

HashCodeEquals面試

2015-08-13 10:29:12

面試面試官

2021-06-28 17:26:15

歸并排序建模

2009-09-28 10:58:45

招聘

2021-09-28 13:42:55

Chrome Devwebsocket網絡協議
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 午夜影院普通用户体验区 | 欧美午夜视频 | 日韩一区二区三区在线 | 欧美精品一区在线 | 天天天插 | 国产剧情一区 | 国产成人综合在线 | 欧美精品a∨在线观看不卡 欧美日韩中文字幕在线播放 | 99爱免费| 国产欧美日韩一区二区三区在线 | 亚洲欧美一区二区三区在线 | 国产精品亚洲综合 | 一区二区三区视频在线 | 亚洲国产欧美在线 | 中文字幕日韩一区 | 亚洲精品www久久久 www.蜜桃av | 日韩三片 | 国产精品久久久久国产a级 欧美日本韩国一区二区 | 国产精品99久久久精品免费观看 | 电影午夜精品一区二区三区 | 久久精品视频在线观看 | 日韩精品一区二区三区 | 性一交一乱一伦视频免费观看 | 日韩中文字幕一区 | 精品免费在线 | 神马久久久久久久久久 | av中文字幕在线播放 | 国产精品成人一区二区三区夜夜夜 | 操皮视频 | 综合久| 国产成人精品久久二区二区91 | 亚洲精品中文在线观看 | 久久精品91久久久久久再现 | 国产在线观看一区二区三区 | 亚洲欧美日韩精品久久亚洲区 | 欧美久久精品 | 中文字幕免费 | 最新中文字幕 | 国产精品色 | 久久av网 | 凹凸日日摸日日碰夜夜 |