fork函數(shù)是創(chuàng)建一個(gè)新的進(jìn)程作為原進(jìn)程的子進(jìn)程,創(chuàng)建的子進(jìn)程和父進(jìn)程存在很多的相似性,首先父子進(jìn)程的虛擬存儲(chǔ)空間的用戶空間是相同的,是將父進(jìn)程的拷貝給子進(jìn)程。同時(shí)父子進(jìn)程對(duì)文件的操作是共享方式。因?yàn)楦高M(jìn)程的文件描述符表被拷貝給了子進(jìn)程(具體的原理參虛擬存儲(chǔ)器的內(nèi)容,私有對(duì)象寫時(shí)拷貝實(shí)現(xiàn)了父子進(jìn)程之間形成相互獨(dú)立的地址空間)。因此父進(jìn)程打開的所有文件描述符都在子進(jìn)程中保存了(每個(gè)進(jìn)程都有獨(dú)立的描述符表)。由于所有的進(jìn)程共享文件表、v-node表,所以父子進(jìn)程的描述符表也是相同的,所以父子進(jìn)程對(duì)文件是以共享的方式存在的。
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<sys/wait.h>
int main()
{
int fd;
char c[3];
/*打開文件foobar.txt,采用的是只讀形式*/
fd = open("foobar.txt",O_RDONLY,0);
if(fork()==0)//子進(jìn)程
{
read(fd,&c,2);/*讀文件的一個(gè)字節(jié)到c中*/
c[2]='\0';
printf("c = %s\n",c);
exit(0);
/*子進(jìn)程結(jié)束*/
}
/*下面是父進(jìn)程的讀操作*/
wait(NULL);
read(fd,&c,2);
c[2]='\0';
printf("c = %s\n",c);
exit(0);
}
其中foobar.txt中的內(nèi)容是foobar。
編譯調(diào)試以后的結(jié)果是:
[gong@Gong-Computer cprogram]$ gcc -g fileshare2.c -o fileshare2
[gong@Gong-Computer cprogram]$ ./fileshare2
c = fo
c = ob
原因分析:由于父子進(jìn)程是以共享的方式控制已經(jīng)打開文件的,因此對(duì)文件的操作也是相互影響的,因此讀寫文件的位置也會(huì)發(fā)生相應(yīng)的改變。父(子)進(jìn)程的文件讀寫位置會(huì)隨著子(父)進(jìn)程的文件讀寫位置改變而改變,因?yàn)榇藭r(shí)改變的是文件表的文件位置項(xiàng),而文件表是所有進(jìn)程共享的,任何一個(gè)進(jìn)程的修改都會(huì)影響到別的進(jìn)程。但是父(子)進(jìn)程對(duì)描述符的修改不會(huì)影響子(父)進(jìn)程的描述符,因?yàn)閏lose(fd)的操作只是改變文件表述符表中的內(nèi)容,而該表是每個(gè)進(jìn)程相互獨(dú)立的,因此不會(huì)改變其他進(jìn)程的表。