linux中的fork()函数以及标准I/O缓冲
1. fork()创建的新进程成为子进程。一次调用,两次返回,子进程的返回值是0,而父进程的返回值是新子进程的进程ID,如果出现错误,fork返回一个负值。
2. 可以通过fork返回的值来判断当前进程是子进程还是父进程。
3. 为什么在父子进程中不同?“其实就相当于链表,进程形成了链表,父进程的fpid(p 意味point)指向子进程的进程id, 因为子进程没有子进程,所以其fpid为0.
4. C程序由正文段、初始化数据段、非初始化数据段、堆和栈组成。
5. 子进程获得父进程的数据段、堆和栈的副本,即复制而并非共享,并不包括正文段,但父子进程共享正文段。
6. !!!子进程和父进程继续执行fork()调用之后的指令。
7. fork()出错可能有两种原因:
1)当前的进程数已经达到了系统规定的上限,这时errno的值被设置为EAGAIN。
2)系统内存不足,这时errno的值被设置为ENOMEM。
创建新进程成功后,系统中出现两个基本完全相同的进程,这两个进程执行没有固定的先后顺序,哪个进程先执行要看系统的进程调度策略。
1)当前的进程数已经达到了系统规定的上限,这时errno的值被设置为EAGAIN。
2)系统内存不足,这时errno的值被设置为ENOMEM。
创建新进程成功后,系统中出现两个基本完全相同的进程,这两个进程执行没有固定的先后顺序,哪个进程先执行要看系统的进程调度策略。
8. 每个进程都有一个独特(互不相同)的进程标识符(process ID),可以通过getpid()函数获得,还有一个记录父进程pid的变量,可以通过getppid()函数获得变量的值。
9. fork()是把进程当前的情况拷贝一份,执行fork时,进程已经执行完了前面的步伐;fork只拷贝下一个要执行的代码到新的进程。
for (int i = 0; i < N; ++i) { pid_t fpid=fork(); if(fpid==0) printf("son\n"); else printf("father\n"); }
10. 这种N次循环的情况,执行printf函数的次数为2*(1+2+4+……+2N-1)次,创建的子进程数为1+2+4+……+2N-1个
11. 想测一下一个程序中到底创建了几个子进程,最好的方法就是调用printf函数打印该进程的pid,也即调用printf("%d\n",getpid());或者通过printf("+\n");来判断产生了几个进程。
1. 标准I/O库提供缓冲的目的是尽可能的减少使用read和write调用的次数。
2. 标准I/O缓冲有三种:全缓冲、行缓冲和不带缓冲。行缓冲由换行符进行冲洗。
3. printf某些内容时,操作系统仅仅是把该内容放到了stdout的缓冲队列里了,并没有实际的写到屏幕上。但是,只要看到有\n 则会立即刷新stdout,因此就马上能够打印了。
4. 运行了printf("fork!")后,“fork!”仅仅被放到了缓冲里,程序运行到fork时缓冲里面的“fork!” 被子进程复制过去了。因此在子进程度stdout缓冲里面就也有了fork! 。所以你最后看到fork!被打印了2次。
4. 运行了printf("fork!")后,“fork!”仅仅被放到了缓冲里,程序运行到fork时缓冲里面的“fork!” 被子进程复制过去了。因此在子进程度stdout缓冲里面就也有了fork! 。所以你最后看到fork!被打印了2次。
5. 运行printf("fork! \n")后,“fork!”被立即打印到了屏幕上,之后fork到的子进程里的stdout缓冲里不会有fork! 内容。因此你看到的结果会是fork! 被printf了1次!!!!