1、 fork进程创建
代码:
#include <stdio.h> #include <string.h> #include <stdlib.h> #include <fcntl.h> #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> #include <errno.h> #include <math.h> int main(int argc, char *argv[]) { pid_t child; //create child process if((child = fork()) == -1) { printf("Fork Error: %s\n", strerror(errno)); exit(1); } else { if(child == 0) { printf("I am the child : %d\n", getpid()); exit(0); } else { printf("I am the father : %d\n", getpid()); return 0; } } return 0; }
Makefile:
CC = gcc CURTDIR = $(shell pwd) TARGET = myfork %.o:%.c $(CC)-c $(EXTRAFLAGS) $< -o $@ %.o:%.S $(CC)-c $(EXTRAFLAGS) $< -o $@ .PHONY: all clean $(TARGET): $(TARGET).o $(CC) -o $@ $^ clean: rm-rf $(TARGET) $(TARGET).o
运行结果:
eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.8$make gcc -c myfork.c -o myfork.o gcc -o myfork myfork.o eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.8$ls Makefile myfork myfork.c myfork.o eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.8$./myfork I am the father : 8694 eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.8$I am the child : 8695
总结:fork函数创建子进程后,父子进程是独立、同时运行的,并没有先后顺序之分。
2、vfork进程创建
代码:
#include <stdio.h> #include <string.h> #include <stdlib.h> #include <fcntl.h> #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> #include <errno.h> #include <math.h> int main(int argc, char *argv[]) { pid_t child; //create child process if((child = vfork()) == -1) { printf("Fork Error: %s\n", strerror(errno)); exit(1); } else { if(child == 0) { sleep(1); printf("I am the child :%d\n", getpid()); exit(0); } else { printf("I am the father : %d\n", getpid()); return 0; } } return 0; }
Makefile:
CC = gcc CURTDIR = $(shell pwd) TARGET = myvfork %.o:%.c $(CC)-c $(EXTRAFLAGS) $< -o $@ %.o:%.S $(CC)-c $(EXTRAFLAGS) $< -o $@ .PHONY: all clean $(TARGET): $(TARGET).o $(CC) -o $@ $^ clean: rm-rf $(TARGET) $(TARGET).o
运行结果:
eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.9$make gcc -c myvfork.c -o myvfork.o gcc -o myvfork myvfork.o eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.9$ls Makefile myvfork myvfork.c myvfork.o eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.9$./myvfork I am the child : 8870 I am the father : 8869
总结:调用fork函数产生的父子进程的运行顺序是不定的,而调用vfork函数产生的父子进程必定是子进程先运行完,父进程再运行的。
3、 exec函数族
代码:
#include <stdio.h> #include <string.h> #include <stdlib.h> #include <fcntl.h> #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> #include <errno.h> #include <math.h> int main(int argc, char *argv[]) { if(argc < 2) { perror("you haven't input the filename, please try again!\n"); exit(EXIT_FAILURE); } if(execl("./file_create", "file_creat", argv[1],NULL) < 0) perror("execl error!\n"); return 0; }
Makefile:
CC = gcc CURTDIR = $(shell pwd) TARGET = myexec %.o:%.c $(CC)-c $(EXTRAFLAGS) $< -o $@ %.o:%.S $(CC)-c $(EXTRAFLAGS) $< -o $@ .PHONY: all clean $(TARGET): $(TARGET).o $(CC) -o $@ $^ clean: rm-rf $(TARGET) $(TARGET).o
运行结果:
eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.10$make gcc -c myexec.c -o myexec.o gcc -o myexec myexec.o eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.10$ls file_create Makefile myexec myexec.c myexec.o eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.10$./myexec file create file file success! eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.10$ls file file_create Makefile myexec myexec.c myexec.o
总结:exec函数族会在一个进程中启动另一个进程执行。并用它来取代原调用进程的数据段、代码段和堆栈段。在执行完exec函数调用后,原调用进程的内容除了进程号之外,其他全部都被新的进程替换了。
4、 等待进程
代码:
#include <stdio.h> #include <string.h> #include <stdlib.h> #include <fcntl.h> #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> #include <errno.h> #include <math.h> int main(int argc, char *argv[]) { pid_t child; //create child process if((child = vfork()) == -1) { printf("Fork Error: %s\n", strerror(errno)); exit(1); } else { if(child == 0) { printf("the child process is run\n"); sleep(1); printf("I am the child : %d\n", getpid()); exit(0); } else { wait(NULL); printf("the father process is run\n"); printf("I am the father : %d\n", getpid()); return 0; } } return 0; }
Makefile:
CC = gcc CURTDIR = $(shell pwd) TARGET = mywait %.o:%.c $(CC)-c $(EXTRAFLAGS) $< -o $@ %.o:%.S $(CC)-c $(EXTRAFLAGS) $< -o $@ .PHONY: all clean $(TARGET): $(TARGET).o $(CC) -o $@ $^ clean: rm-rf $(TARGET) $(TARGET).o
运行结果:
eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.11$make gcc -c mywait.c -o mywait.o gcc -o mywait mywait.o eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.11$ls Makefile mywait mywait.c mywait.o eastmoon@eastmoon-virtual-machine:~/work/guoqian/1/1.11$./mywait the child process is run I am the child : 9275 the father process is run I am the father : 9274
结论:用fork创建子进程后,父子进程的执行顺序fork函数不能确定。我们可以用wait函数和waitpid函数使父进程阻塞等待子进程退出的性质,来确保子进程先结束。