linux的進程
Linux內核在系統啟動的最后階段會啟動init進程。Linux系統的進程之間存在著明顯的繼承關系,所有的進程都是pid為1的init進程的后代。
其他一些操作系統在創建進程時,首先在地址空間里創建進程,讀入可執行文件,最后開始執行。Linux是類Unix的操作系統,關于進程創建,它不同于前面那些操作系統,而是定義了fork()和exec()兩組函數。這里以fork()函數為例介紹題主所說的“寫時拷貝”。
linux創建進程的資源
fork()函數通過拷貝父進程創建子進程,子進程與父進程的區別僅僅在于pid,ppid和一些資源的統計量,比如掛起的信號等。在早期,fork()函數會將父進程的所有其他資源都復制給子進程。這種設計過于簡單粗暴,因為子進程也許并不需要父進程的資源,如果子進程被創建后,轉而執行和之前毫不相關的工作,那之前拷貝資源的開銷就浪費了,一點意義也沒有。
為了解決上面提到的可能會出現浪費的問題,“寫時拷貝”的概念就被提出了。寫時拷貝是一種可以推遲甚至免去拷貝數據的技術。子進程被創建后,系統將父進程的資源以只讀的方式共享給子進程,這樣子進程能夠使用原本應該拷貝給子進程的數據,而同時又不會“污染”父進程。
這樣一來,如果子進程只需要讀取父進程數據,或者不需要使用父進程的數據,那么拷貝就免去了。如果子進程需要
實例
結合上面這兩點,就是“寫時拷貝”的含義了,下面給出demo:
char*buf=(char*)malloc(100*1024*1024);
intpid=fork();
if(pid==0){
printf("childexit\n");
free(buf);
exit(0);
}else{
wait(&status);
free(buf);
exit(0);
}
對于上面這種情況,因為子進程沒有用到父進程的buf,所以系統就免去了拷貝buf100MB的開銷,提升了效率。
char*buf=(char*)malloc(100*1024*1024);
intpid=fork();
if(pid==0){
buf[0]=1;
printf("childexit\n");
free(buf);
exit(0);
}else{
wait(&status);
free(buf);
exit(0);
}
而對于上面這種情況,因為子進程需要