【问题标题】:C fork() causing unexpected function calls that aren't specifiedC fork() 导致未指定的意外函数调用
【发布时间】:2018-05-16 01:00:45
【问题描述】:

我的第一个 Stack Overflow 帖子,我是该网站的潜伏者,但我真的很想开始问一些我自己的问题!

我正在学习使用 unistd.h 库的 fork() 系统调用。我想测试我正在学习的内容。我遵循了一本教科书,以获取有关在 UNIX 中创建简单进程的示例。我在 OSX 上这样做。我被设置了一个创建 shell 的任务,我不想得到关于构建 shell 的完整答案,因此为什么我要询问这个特定的系统调用,而不是整个程序。

代码自行运行完全正常,并给出了预期的输出。

这是正常工作的代码:

#include <stdio.h>
#include <unistd.h>

int main() {
    pid_t pid;
    pid = fork();
    if(pid == 0){
        printf("\nI'm the child\n\n");
    }
    else if(pid > 0){
        printf("\nI'm the parent, child has pid: [%d]\n", pid);
    }
    else{
        printf("ERROR");
    }
    return 0;
}

我的问题是,当我将它放入我的 shell 程序时,根据我在网上找到的内容,它会导致一个 fork 炸弹。

使用命令“test”调用创建此进程的函数。当我运行该函数时,它提供与独立版本相同的输出,但是当我继续并退出时,它似乎在无限循环中再次调用该测试函数。然后我无法杀死这些进程,因为我的计算机滞后,显然是因为正在创建一堆进程,我必须重新启动我的计算机。我已经这样做了几次,由于每次运行时都需要重新启动,因此无法进行测试。

我有一个等待退出命令结束的 while 循环。代码应该是不言自明的,如下所示。我知道很多这是不好的做法,我只是想测试这些概念,从研究中我相信我应该使用 wait() 系统调用,但我想了解为什么我的代码会导致这种情况发生。

完整程序:

main.c

#include "shell.h"

int main() {
    clear();
    printf("Jack's Shell\n");
    shellLoop();
    clear();
    return 0;
}

shell.c

#include "shell.h"

void shellLoop(){
    char command[MAX_STRING_LEN];
    int exit = 1;
    while (exit != 0){
        printf("\njackdewinter$ ");
        scanf("%s", &command);
        exit = commandInterpreter(command);
    }

}

int commandInterpreter(char * cmd){
    char message[MAX_STRING_LEN];
    if(strcmp(cmd, "exit") == 0){ /* Best way to test strings (Not just characters.) */
        return 0;
    }
    else if(strcmp(cmd, "info") == 0){
        info();
        return 1;
    }
    else if(strcmp(cmd, "pwd") == 0){
        shellMessage("Call PWD");
        return 1;
    }
    else if(strcmp(cmd, "cd") == 0){
        shellMessage("Call CD");
        return 1;
    }
    else if(strcmp(cmd, "test") == 0){
        shellMessage("Test Called.");
        test();
        return 1;
    }
    else if(strcmp(cmd, "help") == 0){
        shellMessage("Call HELP");
        return 1;
    }
    else{
        shellMessage("Incorrect command entered.\nType help for assistance and a list of commands.");
        return 1;
    }
}

void info(){
    shellMessage("This shell was created by Jack Dewinter");
}

void test(){
    pid_t pid;
    pid = fork();
    if(pid == 0){
        printf("Child\n");
    }
    else if(pid > 0){
        printf("I'm the parent, child has pid %d\n", pid);
    }
    else{
        printf("ERROR");
    }
}

void shellMessage(char * message){ /* Using this for consistent shell messages. */
    printf("\n%s\n", message);
}

shell.h

#define MAX_STRING_LEN 80

#define clear() printf("\033[H\033[J") /* Terminal escape codes for clearing the console */

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>

void shellLoop();
int commandInterpreter(char * cmd);
void info();
void test();
void shellMessage(char * message);

任何帮助将不胜感激,谢谢!

【问题讨论】:

  • 这不是叉子炸弹,它只是一个错误。子进程应该在完成后exit()。 IE。在if (pid == 0) 块内。最终,该分叉进程应该终止。据我所知,父母没有理由关心它是否/何时这样做,但它仍然应该,或者它也将在其进程中返回命令解释器和选择菜单。
  • 您的孩子和父母都返回您的循环。也不退出。所以现在,你有两个过程。当您在标准输入中输入某些内容时,它会发送给孩子。但如果子进程退出,父进程仍在运行,等待从标准输入读取。如果您多次运行测试,您将有多个实例等待输入。虽然我没有看到一个循环制造炸弹......
  • 这段代码存在缓冲区溢出。您应该将scanf("%s", &amp;command) 替换为scanf("%80s", &amp;command) 或使用fgets
  • @DavidConrad -- 我看不到这会导致缓冲区溢出,除非有人输入了很长的字符串。不过,防止这种情况是一种很好的做法。
  • “有缓冲区溢出”表示有缓冲区溢出...

标签: c linux shell fork system-calls


【解决方案1】:

您正在创建的孩子将像他的父亲一样循环,并且您将让每个循环都有一个新进程。 fork :创建确切的进程映像。

你可以做的是: 1-加入孩子(即父亲等待孩子完成)

2- 或者只是在 printf("Child\n"); 之后添加一个 exit();这将结束孩子,并让父亲继续运行。

【讨论】:

  • 谢谢。我知道它们都将从创建孩子的那一刻开始运行,但是循环不应该再次调用测试函数,除非“测试”作为命令输入,由于某种原因它正在循环并再次启动该函数。
【解决方案2】:

当您使用fork 创建新进程时,子进程会从分叉点继续。因此,您的孩子会打印“Child”,然后它会从 test 返回。所以它继续运行只有父级应该运行的代码。

您在较小的代码中看不到这一点,因为您立即从 main 返回导致子进程退出。

你需要让孩子在完成它的部分后退出。此外,父母应该为孩子wait,这样您就不会留下僵尸进程。

void test(){
    pid_t pid;
    pid = fork();
    if(pid == 0){
        printf("Child\n");
        _exit(0);    // exit child; use _exit instead of exit 
                     //to prevent atexit handlers from being called
    }
    else if(pid > 0){
        printf("I'm the parent, child has pid %d\n", pid);
        // wait for child to finish
        wait(NULL);
    }
    else{
        printf("ERROR");
    }
}

此外,在 shellLoop 中,将变量 exit 重命名为其他名称,因为您要屏蔽库函数 exit

【讨论】:

  • 谢谢,这解决了问题。当我尝试运行程序时,我的计算机不再中断,wait(NULL) 也允许继续执行,否则它会中断。
  • 也感谢关于退出变量的提示,我在阅读本书时快速写了这个,我将完全重写这个,我看到很多人使用这些类型的状态变量程序。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-10-10
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多