虽然我认为我掌握了fork()
,exec()
,wait()
和pid
如何在C中工作,但我还没有找到办法如何运行程序中的个人程序。
这是我的代码:
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h> /* for fork() */
#include<sys/types.h> /* for pid_t */
#include<sys/wait.h> /* fpr wait() */
int main(int argc, char* argv[])
{
char fileName[255];
pid_t pid;
switch (pid = fork()) {
case -1: //Did not fork properly
perror("fork");
break;
case 0: //child
execv(fileName[0],fileName);
puts("Oh my. If this prints, execv() must have failed");
exit(EXIT_FAILURE);
break;
default: //parent
//Infinite Loop
while (1) {
printf(" %s > ", argv[0]);
scanf("%s", fileName); // gets filename
if (fileName[0] == '\0') continue;
printf("\n Entered file: %s",fileName); // prints the fileName
waitpid(pid,0,0); /* wait for child to exit() */
break;
}
}
return 0;
}
我的问题如下:
我想将一个字符串作为输入,我想将其范围限制为255个字符。 char fileName[255]
然后是scanf("%s", fileName);
还有什么路要走?我应该使用getLine()
还是其他功能?
让我们说输入是正确的。如何执行说现有的hello world程序。输入是否会存储在*argv[]
中?我发现在不同的程序中我可以使用
static char *argv[] = { "echo", "Foo is my name." , NULL };
execv("/bin/echo", argv);
为了回应&#34; Foo是我的名字。&#34;。我可以用helloWorld程序做类似的事情吗?
答案 0 :(得分:2)
您将一个字符作为命令名称传递,并将名称字符串作为参数列表的开头传递 - 就好像execv()
的原型是 int execv(char cmd, char *args)
实际原型是:int execv(char *cmd, char **args)
,所以你需要:
char *args[2] = { fileName, 0 };
execv(args[0], args);
我假设您将fileName
设置为某个有意义的值 - 未显示。例如,它可能是"./local_program"
。它将被视为可执行文件的路径名。
如果您想阅读该名称,则可以使用fgets()
或getline()
,但您需要删除换行符:
if (fgets(fileName, sizeof(fileName), stdin) != 0)
{
fileName[strcspn(fileName, "\n")] = '\0';
…as before…
}
或:
char *fileName = 0;
size_t length = 0;
if (getline(&fileName, &length, stdin) != -1) /* Not EOF! */
{
fileName[strcspn(fileName, "\n")] = '\0';
…as before…
}
free(fileName);
strcspn()
的使用避免了特殊情况下过长的命令行,以至于fileName
中没有换行符。请注意,这不会尝试将输入行拆分为命令名和空格中的参数或类似的任何东西。这是shell实现的下一个级别:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
int main(void)
{
char fileName[256];
if (fgets(fileName, sizeof(fileName), stdin) != 0)
{
fileName[strcspn(fileName, "\n")] = '\0';
char *args[129];
char **argv = args;
char *cmd = fileName;
const char *whisp = " \t\f\r\b\n";
/* Ugh — strtok()? OK while only handling white space separators */
char *token;
while ((token = strtok(cmd, whisp)) != 0)
{
*argv++ = token;
cmd = 0;
}
*argv = 0;
execv(args[0], args);
fprintf(stderr, "Oops!\n");
}
return 1;
}
我不需要检查args
数组的溢出,因为输入的256个字符(减去终止空值)不能被分割以产生超过128个单字符参数,每个参数与相邻的参数分开一个空白字符。使用strtok()
是临时的创可贴。一旦你需要处理真正的shell语法(管道,I / O重定向,引用带空格的字符串等),strtok()
就是错误的工具。 (它 - strtok()
- 也是在任何库函数中使用的错误函数。如果必须使用strtok()
,请在Unix上使用POSIX strtok_r()
或在Windows上使用Microsoft的strtok_s()
类似于库代码中的函数。)
答案 1 :(得分:0)
目前,您需要修复代码中的编译错误:
argc
警告(关于未使用的fileName
)是无害的,但错误是真实的,需要修复。
您需要在分叉之前初始化#include<stdio.h>
#include<stdlib.h>
#include<unistd.h> /* for fork() */
#include<sys/types.h> /* for pid_t */
#include<sys/wait.h> /* fpr wait() */
int main(int argc, char* argv[])
{
char fileName[255];
pid_t pid;
//Infinite Loop
while (1) {
printf(" %s > ", argv[0]);
scanf("%s", fileName);
if (fileName[0] == '\0')
break;
printf("\n Entered file: %s\n", fileName);
pid = fork();
switch (pid) {
case -1: //Did not fork properly
perror("fork");
break;
case 0: //child
execl(fileName, fileName, 0);
perror("exec");
exit(EXIT_FAILURE);
break;
default: //parent
waitpid(pid,0,0); /* wait for child to exit() */
break;
}
}
return EXIT_SUCCESS;
}
,否则孩子将无法获得任何有效数据:
.scroll-here
= link_to "Button", example_path, id: "my-button"