我正在开发一个项目,我必须在C中为linux编写一个命令shell。到目前为止,它正在为没有输入的命令工作(即,shell将运行'date'命令和'ls'命令很好。)
但是,需要一些输入的命令,似乎它将每个输入读作一个单独的命令。例如,对于命令:
% gcc -o testFile testFile.c
好像shell正在运行gcc作为自己的命令,然后是-o,然后是testFile和testfile.c,它应该将gcc作为命令,其他三个条目作为输入。
我不明白代码中发生了什么 - 它可能来自于不完全理解execvp函数(我从几个来源读到它并且我仍然认为我不理解它 - 我以为我做到了! )。
execvp调用在函数execute。
中#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include "commandStorage.c"
#define MAX_ARGUMENTS 10
void parseLine(char *command, char **args) {
const char split = '\0';
int i;
char *ptrToken;
while(*command != '\0') {
while ((*command == '\n') || (*command == '\t')
|| (*command == ' ')) {
*(command++) = '\0';
} // end 'while'
*args++ = command;
printf("%s\n", command);
while ((*command != '\n') && (*command != '\t')
&& (*command != '\0') && (*command != ' ')) {
command++;
} // end 'while'
} // end 'while'
*args = '\0';
} // end parseLine
void execute(char **arrayOfPtrs) {
/*
CURRENT BUG:
The function is taking the array of pointers, and executing each
input from the array seperately.
The execvp function is being misused.
*/
pid_t pid;
int waitStatus;
switch (pid = fork()) {
case 0: // Child process
if (execvp(arrayOfPtrs[0], arrayOfPtrs) < 0) {
perror("THE COMMAND FAILED TO EXECUTE!");
break;
} // end 'if
case -1: // Fork failed
perror("THE PROCESS FAILED TO FORK");
break;
default: // Parent process
while ((wait(&waitStatus) != pid)) {};
break;
} // end 'switch'
return;
} // end 'execute
void clearPointerArray(char **args){
while (*args != NULL) {
*(args++) = NULL;
}
}
int main() {
int i;
char command[MAX_STRING_LENGTH]; // unparsed command
pid_t pid;
char *args[MAX_ARGUMENTS]; // Argument vector.
while(1) {
clearPointerArray(args);
printf("%s", "TimsShell--> ");
scanf("%s", command);
parseLine(command, args);
if ((strcmp(args[0], "exit") == 0) || (strcmp(args[0], "quit") == 0)) {
printf("%s\n", "Goodbye!");
return 0;
}
execute(args);
}// while loop
return 0;
} // main
以下是命令行的一些输出:
% gcc -o mainShell mainShell.c
% ./mainShell
TimsShell--> date
date
Fri Feb 14 15:50:28 EST 2014
TimsShell--> ls
ls
change.log doLocalConf.xml license.txt notepad++.exe session.xml testFile.c
commandStorage.c functionList.xml localization plugins shortcuts.xml themes
config.model.xml langs.model.xml mainShell readme.txt stylers.model.xml updater
config.xml langs.xml mainShell.c SciLexer.dll stylers.xml user.manual
TimsShell--> gcc -o testFile testFile.c
gcc
gcc: fatal error: no input files
compilation terminated.
TimsShell--> -o
THE COMMAND FAILED TO EXECUTE!: No such file or directory
TimsShell--> testFile
THE COMMAND FAILED TO EXECUTE!: No such file or directory
TimsShell--> testFile.c
: not found 2: testFile.c:
testFile.c: 3: testFile.c: Syntax error: "(" unexpected
TimsShell--> ^C
答案 0 :(得分:1)
有关发生这种情况的原因,请参阅scanf的手册页。问题在于格式字符串:
%S
匹配一系列非空白字符;下一个指针 必须是一个指向字符数组的指针,该数组足够长以容纳 输入序列和添加的终止空字节('\ 0') 自动。输入字符串在空白处或最大处停止 字段宽度,以先到者为准。
请改为尝试:
scanf("%[^\n]s", command);
getchar();
或
scanf(" %[^\n]s", command);