【发布时间】: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", &command)替换为scanf("%80s", &command)或使用fgets。 -
@DavidConrad -- 我看不到这会导致缓冲区溢出,除非有人输入了很长的字符串。不过,防止这种情况是一种很好的做法。
-
“有缓冲区溢出”表示有缓冲区溢出...
标签: c linux shell fork system-calls