我最近开始使用Linux,所以我的程序只是为了学习。话虽如此,这是我的代码:
#include <stdio.h>
#include <stdlib.h>
int main () {
FILE *fin;
char buf[200];
fin = fopen("provaMake.txt", "r");
if (fin==NULL) {
fin = fopen("/home/giorgio/Desktop/provaMake.txt", "r");
if (fin==NULL) {
printf("finError\n");
exit(1);
}
}
while(fgets(buf, 200, fin) != NULL) {
system(buf);
}
fclose(fin);
}
这是provaMake.txt
文件:
cd /home/giorgio/Desktop/a
rm -f pippo.txt
cd /home/giorgio/Desktop
当我使用以下方法运行它时:
$gcc provaMake.c -o provaMake.o
$./provaMake.o
它没有删除 pippo.txt (在a
里面的目录中Desktop
)。
我在代码中做错了吗?我再说一遍我正在努力学习,所以请向我解释一下(或重定向我在哪里可以学习)。
system()
在 shell 中运行命令。那个shell是一个新进程。它是您的 C 程序的子进程。该cd
命令更改运行它的 shell 进程的当前目录。这是由该外壳的子进程继承的。但它不会影响父进程的当前目录。你打
system()
了三遍。每次启动一个 shell 进程,它运行一个命令,然后结束。cd
在一次调用system()
中不会影响在单独调用中的当前目录system()
。如果您必须使用
system()
,那么解决您问题的最佳方法可能是避免cd
并在一个命令中完成所有操作。但这只有在您可以更改时才有效provaMake.txt
。特别是,这应该作为以下内容provaMake.txt
:如果您必须使用
system()
但不能或不想更改provaMake.txt
,或者如果您对如何使用在同一个 shell 进程中system()
运行文件中的所有命令特别感兴趣,那么您可以传递一个硬编码的命令启动一个新的 shell 并在同一个 shell 中运行该文件中的所有命令。例如,您的 C 程序可以使用以下代码:system()
您将用该命令替换几乎整个程序。不需要缓冲区,因为您的 C 程序本身并不从文件中读取。同样,不需要
fopen()
调用。shell 是从文件中打开、读取和执行命令的东西——这是它的.
内置函数,当使用任何 Bourne 样式的 shell 时都会出现。(system()
使用sh
,这是一个 Bourne 风格的 shell,和 一样bash
。)请注意,如果您的 C 程序随后依赖或以其他方式观察其当前工作目录,则该
cd
命令仍然不会受到运行命令的影响(就像在您的原始代码中一样),因为它们仅在 shell 中运行。出于学习目的,有时以人为的方式做事是合理的,因此我不想告诉您不要编写
system()
用于从文件中逐个运行所有 shell 命令的 C 程序。但是,如果您的实际目标只是运行文件中的所有命令,则不需要 C 程序。你可以只使用你的外壳:如果你想在你当前的shell 进程中运行命令,这样它的环境就会受到影响——例如,
cd
命令的效果在之后仍然存在——那么你可以运行:如果要在新的shell 中运行命令,以便不影响当前 shell 的环境,则将文件名作为命令行传递给 shell:
如果您正在使用或想要使用的 shell 不是 bash,则可以调整该命令。例如:
您可能希望命令列表本身是可执行的。为此,请在开头添加类似这样的内容(根据应使用的 shell 进行调整):
并通过运行以下命令更改其权限,使其可执行:
替换
filename
为实际的文件名,例如/home/giorgio/Desktop/provaMake.txt
.然后你可以像任何程序一样运行该文件:
特别是在最后一种情况下,使用
.txt
后缀命名文件是不寻常的。通常,内容将使用.
内置命令运行的文件通常使用.sh
后缀命名,而将标记为可执行并作为脚本运行的文件使用不带后缀命名。(不过,就 shell 和操作系统而言,文件名并不重要。)我不知道为什么它对你不起作用,当我尝试类似的东西时它起作用了。
无论如何,添加一些
printf
语句对于理解程序在做什么非常有帮助。例如,尝试在system
调用前面的 while 循环中添加这一行:然后再次编译并运行并查看输出,希望这可以帮助您了解正在发生的事情。