给出如下C程序,文件名为ex_vfork.c,请分析在Linux下程序运行的结果,并说明原因。 1 #include <sys/types.h> 2 #include <unistd.h> 3 #include <stdio.h> 4 #include <stdlib.h> 5 int main() 6 { 7 pid_t pid; 8 char *message; 9 int n = 2; 10 11 printf("fork program starting "); 12 pid = vfork(); 13 switch(pid) 14 { 15 case -1: 16 perror("fork failed"); 17 exit(1); 18 case 0: 19 message = "This is the child"; 20 n = 6; 21 break; 22 default: 23 message = "This is the parent"; 24 n++; 25 break; 26 } 27 28 for(; n > 0; n--) { 29 puts(message); 30 sleep(1); 31 } 32 33 exit(0); 34 }

时间: 2023-12-10 19:36:59 浏览: 19
在Linux下,该程序的运行结果是输出如下内容: ``` fork program starting This is the child This is the child This is the child This is the child This is the child This is the child ``` 该程序使用了vfork()系统调用创建进程,与fork()不同的是,vfork()不会像fork()一样复制父进程的内存空间,而是直接与父进程共享同一片内存空间。因此,子进程在修改变量n的值时,也会影响到父进程的n的值。 在程序中,当pid=vfork()执行后,创建了一个新的子进程。在子进程中,变量message被赋值为"This is the child",变量n被赋值为6。在父进程中,变量message被赋值为"This is the parent",变量n被加1。在for循环中,子进程和父进程都会输出它们各自的message变量的值,并且都会执行n次循环,每次循环输出一次message,然后睡眠1秒钟。 由于vfork()并不会复制父进程的内存空间,因此当子进程修改变量n的值时,也会影响到父进程中变量n的值,导致父进程中的n也被修改为6。因此,for循环在子进程和父进程中的循环次数都是6次,输出了"This is the child"六次。在程序结束时,子进程和父进程都会调用exit()函数退出。
相关问题

给出具体实例,通过调用fork和vfork函数,演示在Linux 平台如何创建一个新进程

好的,下面是一个简单的示例代码,演示如何通过调用fork和vfork函数,在Linux平台上创建一个新进程: ``` #include <stdio.h> #include <unistd.h> #include <sys/types.h> int main() { pid_t pid_fork, pid_vfork; int i = 0; // 使用fork函数创建一个新进程 pid_fork = fork(); if (pid_fork < 0) { fprintf(stderr, "Fork failed"); return 1; } else if (pid_fork == 0) { // 子进程 printf("This is child process created by fork. i=%d\n", ++i); } else { // 父进程 printf("This is parent process. Child pid=%d, i=%d\n", pid_fork, i); } // 使用vfork函数创建一个新进程 pid_vfork = vfork(); if (pid_vfork < 0) { fprintf(stderr, "vfork failed"); return 1; } else if (pid_vfork == 0) { // 子进程 printf("This is child process created by vfork. i=%d\n", ++i); _exit(0); } else { // 父进程 printf("This is parent process. Child pid=%d, i=%d\n", pid_vfork, i); } return 0; } ``` 在上面的示例代码中,我们首先使用fork函数创建了一个新进程,然后在父进程和子进程中打印出不同的信息。接着,我们使用vfork函数创建了另一个新进程,同样在父进程和子进程中打印出不同的信息。需要注意的是,在使用vfork函数创建子进程后,子进程必须要调用_exit函数或者exec函数,否则可能会引起一些不可预知的问题。 当我们编译并运行上述代码时,可以得到如下输出结果: ``` This is parent process. Child pid=12345, i=0 This is child process created by fork. i=1 This is child process created by vfork. i=2 ``` 可以看到,我们通过调用fork和vfork函数,成功地创建了两个新进程,并在父进程和子进程中打印出了不同的信息。

CLONE_VM|CLONE_VFORK|SIGCHLD

CLONE_VM、CLONE_VFORK、SIGCHLD都是Linux系统中的系统调用。 CLONE_VM选项表示创建一个与父进程共享内存空间的新进程,即父进程和子进程共享内存空间,对其中一方的修改对另一方也可见。 CLONE_VFORK选项表示创建一个新的子进程,该子进程与父进程共享内存空间,但是只有在子进程调用execve()或者exit()后才能使父进程继续执行。 SIGCHLD是一个进程状态改变的信号。当一个子进程停止或终止时,内核向其父进程发送SIGCHLD信号。可以通过调用waitpid()函数获取子进程的状态。

相关推荐

#ifndef CONFIG_HAVE_COPY_THREAD_TLS /* For compatibility with architectures that call do_fork directly rather than * using the syscall entry points below. */ long do_fork(unsigned long clone_flags, unsigned long stack_start, unsigned long stack_size, int __user *parent_tidptr, int __user *child_tidptr) { struct kernel_clone_args args = { .flags = (clone_flags & ~CSIGNAL), .pidfd = parent_tidptr, .child_tid = child_tidptr, .parent_tid = parent_tidptr, .exit_signal = (clone_flags & CSIGNAL), .stack = stack_start, .stack_size = stack_size, }; if (!legacy_clone_args_valid(&args)) //1.查找 pid 位图,为子进程分配新的 pid return -EINVAL; return _do_fork(&args); } long _do_fork(struct kernel_clone_args *args) { u64 clone_flags = args->flags; struct completion vfork; struct pid *pid; struct task_struct *p; int trace = 0; long nr; //2.关于进程追踪的设置 if (!(clone_flags & CLONE_UNTRACED)) { if (clone_flags & CLONE_VFORK) trace = PTRACE_EVENT_VFORK; else if (args->exit_signal != SIGCHLD) trace = PTRACE_EVENT_CLONE; else trace = PTRACE_EVENT_FORK; if (likely(!ptrace_event_enabled(current, trace))) trace = 0; } //3.复制进程描述符 p = copy_process(NULL, trace, NUMA_NO_NODE, args); add_latent_entropy(); if (IS_ERR(p)) return PTR_ERR(p); trace_sched_process_fork(current, p); pid = get_task_pid(p, PIDTYPE_PID); nr = pid_vnr(pid); if (clone_flags & CLONE_PARENT_SETTID) put_user(nr, args->parent_tid); if (clone_flags & CLONE_VFORK) { p->vfork_done = &vfork; init_completion(&vfork); get_task_struct(p); } //4.将子进程放在运行队列中父进程的前面 wake_up_new_task(p); /* forking complete and child started to run, tell ptracer */ if (unlikely(trace)) ptrace_event_pid(trace, pid); if (clone_flags & CLONE_VFORK) { //5.如果是 vfork() 的话父进程插入等待队列,挂起父进程直到子进程释放自己的内存地址空间 //(直到子进程结束或者执行新的程序) if (!wait_for_vfork_done(p, &vfork)) ptrace_event_pid(PTRACE_EVENT_VFORK_DONE, pid); } put_pid(pid); return nr; }加上注释

最新推荐

recommend-type

linux_c API函数大全

LINUX C函数库API 1.字符测试篇 15 1.1 15 isalnum(测试字符是否为英文或数字) 15 1.2 15 isalpha (测试字符是否为英文字母) 15 1.3 16 isascii(测试字符是否为ASCII 码字符) 16 1.4 17 iscntrl(测试字符是否...
recommend-type

Linux下C语言编程手册[原创]

在Linux和Unix操作系统环境下,C语言是开发系统级软件、编写高效程序的首选语言。由于其与操作系统底层的紧密联系,C语言提供了丰富的系统调用来进行进程管理、文件操作、内存管理和网络通信等任务。本手册主要聚焦...
recommend-type

Linux系统及编程基础有答案.doc

* Linux 系统通过目录将系统中的所有文件分级、分层组织在一起,形成了 Linux 文件系统的树形层次结构。 * 在 Linux 系统中建立新目录的命令是 mkdir。 * Linux 系统的文件是一个简单的字节序列。 三、进程管理 * ...
recommend-type

创建新进程:fork函数:fork函数干什么? fork函数与vfork函数的区别在哪里?为何在一个fork的子进程分支中使用_exit函数而不使用exit函数?

fork 函数的基本功能可以通过以下示例程序来说明: ```c pid_t pid; switch (pid = fork()) { case -1: /* fork 函数失败 */ perror("The fork failed!"); break; case 0: /* 子进程 */ /* 这里,我们是孩子...
recommend-type

合信TP-i系列HMI触摸屏CAD图.zip

合信TP-i系列HMI触摸屏CAD图
recommend-type

BSC关键绩效财务与客户指标详解

BSC(Balanced Scorecard,平衡计分卡)是一种战略绩效管理系统,它将企业的绩效评估从传统的财务维度扩展到非财务领域,以提供更全面、深入的业绩衡量。在提供的文档中,BSC绩效考核指标主要分为两大类:财务类和客户类。 1. 财务类指标: - 部门费用的实际与预算比较:如项目研究开发费用、课题费用、招聘费用、培训费用和新产品研发费用,均通过实际支出与计划预算的百分比来衡量,这反映了部门在成本控制上的效率。 - 经营利润指标:如承保利润、赔付率和理赔统计,这些涉及保险公司的核心盈利能力和风险管理水平。 - 人力成本和保费收益:如人力成本与计划的比例,以及标准保费、附加佣金、续期推动费用等与预算的对比,评估业务运营和盈利能力。 - 财务效率:包括管理费用、销售费用和投资回报率,如净投资收益率、销售目标达成率等,反映公司的财务健康状况和经营效率。 2. 客户类指标: - 客户满意度:通过包装水平客户满意度调研,了解产品和服务的质量和客户体验。 - 市场表现:通过市场销售月报和市场份额,衡量公司在市场中的竞争地位和销售业绩。 - 服务指标:如新契约标保完成度、续保率和出租率,体现客户服务质量和客户忠诚度。 - 品牌和市场知名度:通过问卷调查、公众媒体反馈和总公司级评价来评估品牌影响力和市场认知度。 BSC绩效考核指标旨在确保企业的战略目标与财务和非财务目标的平衡,通过量化这些关键指标,帮助管理层做出决策,优化资源配置,并驱动组织的整体业绩提升。同时,这份指标汇总文档强调了财务稳健性和客户满意度的重要性,体现了现代企业对多维度绩效管理的重视。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

【实战演练】俄罗斯方块:实现经典的俄罗斯方块游戏,学习方块生成和行消除逻辑。

![【实战演练】俄罗斯方块:实现经典的俄罗斯方块游戏,学习方块生成和行消除逻辑。](https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/70a49cc62dcc46a491b9f63542110765~tplv-k3u1fbpfcp-zoom-in-crop-mark:1512:0:0:0.awebp) # 1. 俄罗斯方块游戏概述** 俄罗斯方块是一款经典的益智游戏,由阿列克谢·帕基特诺夫于1984年发明。游戏目标是通过控制不断下落的方块,排列成水平线,消除它们并获得分数。俄罗斯方块风靡全球,成为有史以来最受欢迎的视频游戏之一。 # 2.
recommend-type

卷积神经网络实现手势识别程序

卷积神经网络(Convolutional Neural Network, CNN)在手势识别中是一种非常有效的机器学习模型。CNN特别适用于处理图像数据,因为它能够自动提取和学习局部特征,这对于像手势这样的空间模式识别非常重要。以下是使用CNN实现手势识别的基本步骤: 1. **输入数据准备**:首先,你需要收集或获取一组带有标签的手势图像,作为训练和测试数据集。 2. **数据预处理**:对图像进行标准化、裁剪、大小调整等操作,以便于网络输入。 3. **卷积层(Convolutional Layer)**:这是CNN的核心部分,通过一系列可学习的滤波器(卷积核)对输入图像进行卷积,以
recommend-type

绘制企业战略地图:从财务到客户价值的六步法

"BSC资料.pdf" 战略地图是一种战略管理工具,它帮助企业将战略目标可视化,确保所有部门和员工的工作都与公司的整体战略方向保持一致。战略地图的核心内容包括四个相互关联的视角:财务、客户、内部流程和学习与成长。 1. **财务视角**:这是战略地图的最终目标,通常表现为股东价值的提升。例如,股东期望五年后的销售收入达到五亿元,而目前只有一亿元,那么四亿元的差距就是企业的总体目标。 2. **客户视角**:为了实现财务目标,需要明确客户价值主张。企业可以通过提供最低总成本、产品创新、全面解决方案或系统锁定等方式吸引和保留客户,以实现销售额的增长。 3. **内部流程视角**:确定关键流程以支持客户价值主张和财务目标的实现。主要流程可能包括运营管理、客户管理、创新和社会责任等,每个流程都需要有明确的短期、中期和长期目标。 4. **学习与成长视角**:评估和提升企业的人力资本、信息资本和组织资本,确保这些无形资产能够支持内部流程的优化和战略目标的达成。 绘制战略地图的六个步骤: 1. **确定股东价值差距**:识别与股东期望之间的差距。 2. **调整客户价值主张**:分析客户并调整策略以满足他们的需求。 3. **设定价值提升时间表**:规划各阶段的目标以逐步缩小差距。 4. **确定战略主题**:识别关键内部流程并设定目标。 5. **提升战略准备度**:评估并提升无形资产的战略准备度。 6. **制定行动方案**:根据战略地图制定具体行动计划,分配资源和预算。 战略地图的有效性主要取决于两个要素: 1. **KPI的数量及分布比例**:一个有效的战略地图通常包含20个左右的指标,且在四个视角之间有均衡的分布,如财务20%,客户20%,内部流程40%。 2. **KPI的性质比例**:指标应涵盖财务、客户、内部流程和学习与成长等各个方面,以全面反映组织的绩效。 战略地图不仅帮助管理层清晰传达战略意图,也使员工能更好地理解自己的工作如何对公司整体目标产生贡献,从而提高执行力和组织协同性。