Реализация команды 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(" %s\n", myfile->d_name);
}
closedir(mydir);
}
Это мои результаты после выполнения кода:
[[email protected] ~]# ./a.out Downloads
4096 ..
4096 hw22.c
4096 ankur.txt
4096 .
4096 destination.txt
Вот правильные размеры:
[[email protected] ~]# 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
Кто-нибудь может указать на мою ошибку.
Спасибо,
Анкур
Ответы
Ответ 1
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;
Ответ 2
Это окончательный код, который я должен работать для всех, кого это интересует. Он печатает правильные размеры файлов. Кредит отправляется ассерту и муксу для ответа, просто скомпоновав код. Ввод, который я получил для работы, это "./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);
}
Ответ 3
Я считаю, что вы заметите, что если вы ./a.out .
, вы получите ожидаемое поведение.
У вас немного тонкая ошибка, наблюдаемая, если вы посмотрите код возврата вашего вызова на stat(2)
.
Основная ошибка: dirent
, возвращаемая readdir(2)
(myfile
в вашем коде), будет иметь d_name
относительно mydir
. Сначала ваш код будет stat
..
, и поэтому mystat
будет содержать достоверные данные для ..
, тогда все последующие вызовы на stat(2)
не будут выполнены, возвращая -1
, которые вы не проверяете, поэтому mystat
не будет изменен, и вы напечатаете st_size
для старого значения, т.е. из ..
.
Ответ 4
Проблема в том, что когда вы stat("ankur.txt", &mystat)
, вы не работаете с файлом "Downloads/ankur.txt"
. Скорее всего, stat()
терпит неудачу; в качестве альтернативы, он сообщает о другом файле.
Следовательно, вам нужно посмотреть, поддерживает ли ваша система fstatat()
- новое в POSIX 2008 - или устраивает префикс имени файл с именем каталога.
Ответ 5
или, может быть, просто работает система ( "ls -al" )!