Реализация команды ls-al в C
в рамках задания одного из моих классов я должен написать программу на 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(" %sn", 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
может кто-нибудь указать на мою ошибку.
спасибо,
Анкур
5 ответов
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;
Это окончательный код, который я должен работать для всех желающих. Он печатает правильные размеры файлов. Заслуга Аскера и мукса в том, что они ответили, просто собрав код. Input I got this to work for is"./главный. ".
#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);
}
я думаю, вы это заметите, если вы ./a.out .
вы получите поведение, которое вы ожидаете.
у вас есть немного тонкая ошибка, наблюдаемая, если вы изучите код возврата вашего вызова stat(2)
.
основная ошибка:dirent
s возвращается readdir(2)
(the myfile
в коде) будет d_name
по отношению к mydir
. Ваш код stat
..
во-первых, преуспеть, и так mystat
будет содержать допустимые данные для ..
, затем все последующие вызовы stat(2)
потерпит неудачу, возвращаясь -1
, которые вы не проверяете, так что mystat
не будет изменен, и вы будете печатать st_size
для старого значения, т. е. ..
.
беда в том, что когда вы stat("ankur.txt", &mystat)
, вы не работаете над файлом "Downloads/ankur.txt"
. Скорее всего,stat()
не удается; в качестве альтернативы, он сообщает о другом файле.
следовательно, вам нужно посмотреть, поддерживает ли ваша система fstatat()
- new в POSIX 2008 - или организовать префикс имени файла с именем каталога.