我有一个用C编写的shell程序,当我尝试运行命令“ls |”时(只是ls和一个没有任何东西的管道),我的程序永远循环。您可以忽略modePtr ...此解析器所做的是从用户输入中获取一串字符,因此它正在解析命令“ls |”我猜这个问题是因为没有一个“情况”,一个空管命令会陷入,导致无限循环?有什么建议吗?
int parse(char *inputString, char *cmdArgv[], char **supplementPtr, int *modePtr)
{
int cmdArgc = 0, terminate = 0;
char *srcPtr = inputString;
//printf("parse fun%sends", inputString);
while(*srcPtr != '\0' && terminate == 0)
{
*cmdArgv = srcPtr;
cmdArgc++;
//printf("parse fun2%sends", *cmdArgv);
while(*srcPtr != ' ' && *srcPtr != '\t' && *srcPtr != '\0' && *srcPtr != '\n' && terminate == 0)
{
// Handles redirection/pipes
switch(*srcPtr)
{
// Background mode
case '&':
*modePtr = BACKGROUND;
break;
// Output mode
case '>':
*modePtr = OUTPUT_REDIRECTION;
*cmdArgv = '\0';
srcPtr++;
if(*srcPtr == '>')
{
*modePtr = OUTPUT_APP;
srcPtr++;
}
while(*srcPtr == ' ' || *srcPtr == '\t' || *srcPtr == '\0')
srcPtr++;
*supplementPtr = srcPtr;
chop(*supplementPtr);
terminate = 1;
break;
// Input mode
case '<':
*modePtr = INPUT_REDIRECTION;
*cmdArgv = '\0';
srcPtr++;
while(*srcPtr == ' ' || *srcPtr == '\t' || *srcPtr == '\0')
srcPtr++;
*supplementPtr = srcPtr;
chop(*supplementPtr);
terminate = 1;
break;
// Pipe mode
case '|':
*modePtr = PIPELINE;
*cmdArgv = '\0';
srcPtr++;
while(*srcPtr == ' ' || *srcPtr == '\t' || *srcPtr == '\0')
srcPtr++;
*supplementPtr = srcPtr;
//chop(*supplementPtr);
terminate = 1;
break;
}
srcPtr++;
}
// Process commands when these occur
while((*srcPtr == ' ' || *srcPtr == '\t' || *srcPtr == '\n') && terminate == 0)
{
*srcPtr = '\0';
srcPtr++;
}
cmdArgv++;
}
/*srcPtr++;
*srcPtr = '\0';
destPtr--;*/
*cmdArgv = '\0';
return cmdArgc;
}
答案 0 :(得分:1)
如果在普通shell中尝试ls |
,它会在尝试启动任何内容之前提示您输入其余的管道(另一个命令)。没有命令读取输出的管道是无意义的。
这个成语(在代码中重复了几次)被打破了:
srcPtr++;
while (*srcPtr == ' ' || *srcPtr == '\t' || *srcPtr == '\0')
srcPtr++;
你不能跳过字符串末尾的空'\0'
!你会立即陷入'未定义的行为'。您需要查看循环逻辑,可能是:
srcPtr++;
while (*srcPtr == ' ' || *srcPtr == '\t')
srcPtr++;
if (*srcPtr == '\0')
...no more data in string...
答案 1 :(得分:1)
您的问题可能过于复杂。如果你想让一个程序读取用它传送的列表项......
#define MAXLINELEN 1000
#include <stdio.h>
#include <stdlib.h>
int main (int argc, char *argv[]) {
char line[MAXLINELEN];
FILE *fpin;
fpin=stdin;
while(fgets(line,MAXLINELEN,fpin)!=NULL) printf("%s",line);
}
...然后您可以使用strtok()解析每行上的项目,或者如果您使用“ls -1”,则每个列表项目都在唯一的行上。
希望这有帮助。