linux fork

貌似是360的笔试题;

   

#include <stdio.h>

#include <stdlib.h>

   

int main() {

int i;

for (i = 0; i < 4; i++) {

fork();

printf(" - \n");

}

}

   

icode0410@ubuntu:~/Documents/code/linux/process$ gcc forks.c && ./a.out

-

-

-

-

-

-

-

-

-

-

-

-

icode0410@ubuntu:~/Documents/code/linux/process$ -

-

-

-

-

-

-

-

-

-

-

-

-

-

-

-

-

-

   

总个数为:2 + 4 + 8 + 16 = 30;

   

|

fork

/+1 \+1

fork fork

/+1 \+1 /+1 \+1

fork ….

/+1 \+1

fork

/+1 \+1

   

如果将printf(" - \n"); 给成printf(" - ");呢?

icode0410@ubuntu:~/Documents/code/linux/process$ gcc forks.c && ./a.out

- - - - icode0410@ubuntu:~/Documents/code/linux/process$ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -

   

总个数为64;

   

通过上面的图可知,最后实际的进程数为16个;又因为没有\n导致printf的缓冲区在进程结束时在会flush,而此时,每个进程的buf都有4个-,所以会打印64个-。

   

   

### Linux 中 `fork` 函数的使用说明 #### 一、函数定义与返回值解释 `fork()` 是 Linux 系统中的一个重要系统调用,其功能是创建一个新的子进程。新创建的子进程几乎完全复制父进程的状态,包括内存空间、文件描述符等[^3]。 该函数的声明如下所示: ```c #include <sys/types.h> #include <unistd.h> pid_t fork(void); ``` 当成功调用时,`fork()` 的行为取决于当前运行的是哪个进程: - **对于父进程**:`fork()` 返回正值,表示新创建的子进程 ID (PID)[^4]。 - **对于子进程**:`fork()` 返回 0,表明这是由 `fork()` 创建的新进程实例。 - 如果发生错误,则会向父进程返回负值 `-1`,并设置全局变量 `errno` 来指示具体原因。 --- #### 二、典型的应用场景 `fork()` 常被用来实现多任务处理或者构建服务端程序,在这些情况下它能够有效地支持并发操作和资源隔离需求: 1. 并发执行多个独立的任务; 2. 启动守护进程或将某些长期运行的服务置于后台工作模式下; 3. 实现父子进程之间的基本消息传递机制——尽管更复杂的 IPC 需要额外的技术手段来完成[^2]。 下面通过几个具体的例子进一步阐述如何实际运用此 API 完成上述目标之一即简单的并发计算任务分配给不同的 CPU 核心去完成从而提高整体效率。 --- #### 三、代码示例 这里给出一段完整的 C 语言源码展示怎样利用 `fork()` 构建两个分支路径分别打印各自的信息直到结束为止: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> /* For fork() */ #include <sys/types.h> /* For pid_t */ int main(){ pid_t child_pid; printf("Parent process starting...\n"); // Create a new child process. child_pid = fork(); if(child_pid == -1){ perror("Failed to create child"); exit(EXIT_FAILURE); } if(0 == child_pid){ // Child Process Block printf("Child PID=%d, Parent PID=%d\n", getpid(), getppid()); sleep(2); // Simulate some work done by the child. printf("This is output from CHILD PROCESS.\n"); }else{ // Parent Process Block printf("Child PID=%d created successfully!\n",child_pid); sleep(1); // Let parent do something before waiting on its children. printf("Waiting for child process %d to finish...",child_pid); wait(NULL); // Wait until any one of my direct descendants terminates. printf("\t...done! All processes have finished now.\n"); } return EXIT_SUCCESS; } ``` 在这个例子中可以看到我们先检查了 `fork()` 是否失败;如果没有问题就区分是在父亲还是儿子那里继续往下走逻辑链路。注意每次都要记得关闭不再使用的文件句柄以免泄漏资源! --- ### 总结 综上所述,掌握了 `fork()` 不仅可以加深对操作系统底层运作的理解还能帮助开发者设计更加灵活高效的软件架构方案。不过需要注意的是由于涉及到上下文切换开销较大因此不建议频繁调用除非确实有必要这样做才行哦!
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值