一、粉絲提問
fork出的進程的父進程是從哪來的?
粉絲提問,一口君必須滿足
粉絲提問
二、解答
這個問題看上去很簡單,但是要想把進程的父進程相關的所有知識點搞清楚,還是有點難度的,下面我們稍微拓展下,分幾點來講解這個知識點。
1. 如何查看進程ID
每個linux進程都一定有一個唯一的數字標識符,稱為進程ID(process ID),進程ID總是一非負整數,它的父進程叫PPID。
查看進程ID命令:
ps -ef
查看進程
也可以使用函數來獲得進程ID:
#include <sys/types.h>
#include <unistd.h>
pid_t getpid(void); 返回:調用進程的進程ID
pid_t getppid(void); 返回:調用進程的父進程ID
2. 第一個進程init
Linux內核啟動之后,會創建第一個用戶級進程init,由上圖可知, init 進程 (pid=1) 是除了 idle 進程 (pid=0,也就是 init_task) 之外另一個比較特殊的進程,它是 Linux 內核開始建立起進程概念時第一個通過 kernel_thread 產生的進程,其開始在內核態執行,然后通過一個系統調用,開始執行用戶空間的 / sbin/init 程序。
3. fork函數
創建一個進程很簡單,先來認識一下fork函數:
#include <sys/types.h>
#include <unistd.h>
pid_t fork(void);
返回:子進程中為0,父進程中為子進程I D,出錯為-1
由fork創建的新進程被稱為子進程( child process)。該函數被調用一次,但返回兩次,兩次返回的區別是子進程的返回值是0,而父進程的返回值則是子進程的進程ID。
一般來說,在f o r k之后是父進程先執行還是子進程先執行是不確定的。這取決于內核所使用的調度算法。
「舉例:」
#include <unistd.h>
int main()
{
pid_t pid;
if ((pid = fork()) == -1) {
perror("fork");
return -1;
} else if (pid == 0) {
this is child process
printf("The return value is %d In child process!! My PID is %d, My PPID is %d", pid,getpid(), getppid());
} else {
this is parent
printf("The return value is %d In parent process!! My PID is %d, My PPID is %d", pid,getpid(), getppid());
}
return 0;
}
執行結果:
fork
由上圖可知,fork被調用了一次,返回了兩次。
【拓展】
使用fork函數得到的子進程是父進程的處繼承了整個進程的地址空間,包括:「進程上下文、進程堆棧、內存信息、打開的文件描述符、信號控制設置、進程優先級、進程組號、當前工作目錄、根目錄、資源限制、控制終端」等。
fork
fork出的子進程會集成父進程的文件描述符,如果讀寫文件,父子進程之間會互相影響。
4. ./run 運行的程序父進程是誰?
我們來編寫一個例子:
int main(int argc, const char *argv[])
{
while(1);
return 0;
}
編譯執行
gcc test.c
./a.out
例子很簡單,就是創建一個死循環的進程。
ps -ef 查看進程ID
執行結果
由上圖可知,a.out進程的進程ID是2991,父進程ID是2675,即進程bash:
2665
bash的父進程是gnome-terminal,所以大家應該明白,我們打開1個Linux終端,其實就是啟動了1個gnome-terminal進程。我們在這個終端上執行./a.out其實就是利用gnome-terminal的子進程bash通過execve()將創建的子進程裝入a.out:
strace
5. 進程和終端的關系
關于進程和終端的關系可以參考以下文章:
《進程組、會話、控制終端概念,如何創建守護進程?》
6. 父進程死了,子進程怎么辦?
1) 僵尸進程
僵尸進程
如上圖所示,
父進程Process A創建子進程Process B,當子進程退出時會給父進程發送信號SIGCHLD;如果父進程沒有調用wait等待子進程結束,退出狀態丟失,轉換成僵死狀態,子進程會變成一個僵尸進程。
當父進程調用wait,僵尸子進程的結束狀態被提取出來,子進程被刪除,并且wait函數立刻返回。
實例
#include <sys/types.h>
#include <unistd.h>
create a ZOMBIE
* ps -ax | grep a.out to show the zombie
int main()
{
if(fork()) {
//父進程
while(1){
sleep(1);
}
}
//子進程
}
上述程序會保證父進程不退出,一直在while(1)中無限循環,而子進程會立刻退出。
僵尸進程
由上圖可知,父進程是3096,子進程是3097,子進程因為退出后父進程沒有調用wait回收子進程資源,所以子進程3097變成僵尸進程defunct。
ps -ax | grep a.out 查看進程狀態
僵尸進程
2) 孤兒進程
如果父進程退出,并且沒有調用wait函數,它的子進程就變成孤兒進程,會被一個特殊進程繼承,這就是init進程,init 進程會自動清理所有它繼承的僵尸進程。
實例代碼:
#include <sys/types.h>
#include <unistd.h>
int main()
{
if(fork()) {
//父進程
}else{
//子進程
while(1){
sleep(1);
}
}
}
上述程序會保證子進程不退出,一直在while(1)中無限循環,而父進程會立刻退出。
孤兒進程:
孤兒進程
./a.out的父進程ID變成1,所以該子進程被init進程繼承。
三、其他啟動進程的方法
1. exec族函數
fork函數用于創建一個子進程,該子進程幾乎拷貝了父進程的全部內容。exec函數族提供了一種在進程中啟動另一個程序執行的方法。它可以根據指定的文件名或目錄名找到可執行文件,并用它來取代原調用進程的數據段、代碼段和堆棧段。在執行完之后,原調用進程的內容除了進程號外,其他全部都被替換了。可執行文件既可以是二進制文件,也可以是任何Linux下可執行的腳本文件。
每當進程調用一種exec函數時,該進程完全由新程序代換,而新程序從main函數開始執行。Exec并不創建新進程,所以前后進程ID也不會變。Exec只是用另一個新程序替換了當前進程的正文、數據、堆和棧段。
「何時使用?」
當進程認為自己不能再為系統和用戶做出任何貢獻了時就可以調用exec函數,讓自己執行新的程序如果某個進程想同時執行另一個程序,它就可以調用fork函數創建子進程,然后在子進程中調用任何一個exec函數。這樣看起來就好像通過執行應用程序而產生了一個新進程一樣。
「函數原型」
函數原型
2. cron命令
在Linux系統中,計劃任務一般是由cron承擔,我們可以把cron設置為開機時自動啟動。cron啟動后,它會讀取它的所有配置文件(全局性配置文件/etc/crontab,以及每個用戶的計劃任務配置文件),然后cron會根據命令和執行時間來按時來調用度工作任務。
檢查cron是否安裝:
ps -ef | grep cron
croncrontab -u //設定某個用戶的cron服務,一般root用戶在執行這個命令的時候需要此參數
crontab -l //列出某個用戶cron服務的詳細內容
crontab -r //刪除某個用戶的cron服務
crontab -e //編輯某個用戶的cron服務
root查看自己的cron設置:
crontab -u root -l
或者直接看自己名下的任務:
crontab -l
創建任務:
crontab -e
打開默認編輯器編輯后保存退出即可
編輯基本格式 :
*****command
分 時 日 月 周 命令
第1列表示分鐘1~59 每分鐘用*或者 1表示
第2列表示小時1~23(0表示0點)
第3列表示日期1~31
第4列表示月份1~12
第5列標識號星期0~6(0表示星期天)
第6列要運行的命令
如果寫為*, 表示每X
如果想定義間隔,在X后加"/"和間隔的數字
每隔一分鐘打印一下系統時間
1 * * * * date >> ~/t.log //>> means append
3. at
在linux系統如果你想要讓自己設計的備份程序可以自動在某個時間點開始在系統底下運行,而不需要手動來啟動它,又該如何處置呢?
這些例行的工作可能又分為一次性定時工作與循環定時工作,在系統內又是哪些服務在負責?
還有,如果你想要每年在老婆的生日前一天就發出一封信件提醒自己不要忘記,linux系統下該怎么做呢?
但是crontab 主要用來提交不斷循環執行的job, 而at 用來提交一段時間后執行的job(執行完就自動刪除整個job)
「舉例:」
1) 首先檢查atd服務有無開啟在一個指定的時間執行一個指定任務,只能執行一次,且需要開啟atd進程
ps -ef | grep atd查看,
開啟用/etc/init.d/atd start or restart;
開機即啟動則需要運行 chkconfig --level 2345 atd on
2) 定時在11:30am用ls列出當前目錄內容并寫入~/log文件
cd ~
at 11:30am today
at>ls > ~/t.log
at> <EOT> //按Ctl-D退出
審核編輯:符乾江
-
進程
+關注
關注
0文章
206瀏覽量
14181 -
Fork
+關注
關注
0文章
14瀏覽量
3424
發布評論請先 登錄
相關推薦
Linux后臺進程管理詳解

Linux進程狀態詳解

深入解析Linux程序與進程

深入Linux進程管理:提升效率與穩定性的關鍵方法

Python中多線程和多進程的區別

Linux用戶身份與進程權限詳解

深入探討Linux的進程調度器

鴻蒙開發:【進程模型概述】

鴻蒙開發:【進程模型】

評論