web-dev-qa-db-ja.com

Cでのls -alコマンドの実装

クラスの1つからの割り当ての一部として、ls -alコマンドの結果を複製するプログラムをCで作成する必要があります。必要な資料を読みましたが、まだ適切な出力が得られません。ここに私のコードがあります。ファイルサイズとファイル名を印刷するだけですが、印刷のファイルサイズは正しくありません。

コード:

#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

誰でも私の間違いを指摘してください。

ありがとう、

アンクル

16
ankur3000

_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);
....
_

stat()への最後の呼び出しからのリンク_4096_および_._のサイズである_.._を出力する理由について。

注:ディレクトリ名、ファイル名NULLバイト、セパレーターを保持するのに十分な大きさのバッファーを割り当てる必要があります。

_strlen(argv[1]) + NAME_MAX + 2;
_
14
iabdalkader

これは、興味のある人のために働くようになった最終コードです。正しいファイルサイズを印刷します。答えは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);
}
6
hbteibet

./a.out .を実行すると、期待どおりの動作が得られることに気付くと思います。

stat(2)への呼び出しのリターンコードを調べると、わずかな微妙なバグがあります。

基本的な間違い:readdir(2)によって返されるdirents(コード内のmyfile)には、mydirに対してd_nameがあります。コードは最初にstat..成功します。したがって、mystatには..の有効なデータが含まれ、その後stat(2)へのすべての呼び出しは失敗し、-1を返します。確認してください。したがって、mystatは変更されず、古い値、つまりst_size..を出力します。

1
Kristian Glass

問題は、stat("ankur.txt", &mystat)を実行すると、ファイル_"Downloads/ankur.txt"_を操作していないことです。おそらく、stat()は失敗しています。または、別のファイルについてレポートしています。

そのため、システムが fstatat() — POSIX 2008の新機能—をサポートしているかどうかを確認するか、ファイル名の前にディレクトリ名を付けるように手配する必要があります。

0