18

作为我的一门课作业的一部分,我必须用 C 语言编写一个程序来复制 ls -al 命令的结果。我已经阅读了必要的材料,但我仍然没有得到正确的输出。到目前为止,这是我的代码,它只应该打印出文件大小和文件名,但其打印的文件大小不正确。

代码:

#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <dirent.h>

int main(int argc, char* argv[])
{
    DIR *mydir;
    struct dirent *myfile;
    struct stat mystat;

    mydir = opendir(argv[1]);
    while((myfile = readdir(mydir)) != NULL)
    {
        stat(myfile->d_name, &mystat);    
        printf("%d",mystat.st_size);
        printf(" %s\n", myfile->d_name);
    }
    closedir(mydir);
}

这些是我执行代码后的结果:

[root@localhost ~]# ./a.out Downloads
4096 ..
4096 hw22.c
4096 ankur.txt
4096 .
4096 destination.txt

以下是正确的尺寸:

[root@localhost ~]# ls -al Downloads
total 20
drwxr-xr-x.  2 root root 4096 Nov 26 01:35 .
dr-xr-x---. 24 root root 4096 Nov 26 01:29 ..
-rw-r--r--.  1 root root   27 Nov 21 06:32 ankur.txt
-rw-r--r--.  1 root root   38 Nov 21 06:50 destination.txt
-rw-r--r--.  1 root root 1139 Nov 25 23:38 hw22.c

谁能指出我的错误。

谢谢,

安库尔

4

5 回答 5

16

myfile->d_name是文件名而不是路径,所以你需要先将文件名附加到目录"Downloads/file.txt",如果它不是工作目录:

char buf[512];    
while((myfile = readdir(mydir)) != NULL)
{
    sprintf(buf, "%s/%s", argv[1], myfile->d_name);
    stat(buf, &mystat);
....

至于为什么打印4096链接的大小.以及..从最后一次调用到stat().

注意:你应该分配一个足够大的缓冲区来保存目录名、文件名、NULL字节和分隔符,像这样

strlen(argv[1]) + NAME_MAX + 2;
于 2012-11-25T18:49:27.800 回答
7

这是我为任何感兴趣的人工作的最终代码。它打印正确的文件大小。归功于 asker 和 mux 的回答,只需将代码放在一起。我得到这个工作的输入是“./main”。.

#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <dirent.h>

int main(int argc, char* argv[])
{
    DIR *mydir;
    struct dirent *myfile;
    struct stat mystat;

    char buf[512];
    mydir = opendir(argv[1]);
    while((myfile = readdir(mydir)) != NULL)
    {
        sprintf(buf, "%s/%s", argv[1], myfile->d_name);
        stat(buf, &mystat);
        printf("%zu",mystat.st_size);
        printf(" %s\n", myfile->d_name);
    }
    closedir(mydir);
}
于 2015-10-13T00:37:30.597 回答
1

我相信你会观察到,如果你./a.out .会得到你期望的行为。

您有一个稍微微妙的错误,如果您检查对stat(2).

根本错误:(代码中的)dirent返回的 s将与. 您的代码将首先成功,因此将包含有效数据,然后所有后续调用都将失败,返回您不检查的,因此不会被修改,您将打印旧值,即的。readdir(2)myfiled_namemydirstat ..mystat..stat(2)-1mystatst_size..

于 2012-11-25T18:51:47.280 回答
0

麻烦的是,当你stat("ankur.txt", &mystat)工作的时候,你不是在处理文件"Downloads/ankur.txt"。最有可能的stat()是,失败了;或者,它正在报告不同的文件。

因此,您需要查看您的系统是否支持fstatat()(POSIX 2008 中的新功能)或安排在文件名前加上目录名。

于 2012-11-25T18:49:10.547 回答
-7

或者也许只是 system("ls -al") 也可以!

于 2013-11-13T07:06:42.123 回答