ツリー ウォーキングを行うために nftw システム コールを使用する小さなコードを書きました。
int flags =0;
flags = flags | FTW_MOUNT;
int nopenfd = 10;
if( nftw( argv[1], sum_sizes, nopenfd, flags) == -1 )
return EXIT_FAILURE;
このオプションを使用すると、nftw はディレクトリがマウント ポイントである場合にディレクトリをスキャンせず、シンボリック リンクを逆参照します (デフォルトの動作)。
各ファイル nftw で、次の関数を呼び出します。
/*total_size is the sum of each file/directory/link*/
long long int total_size, total_real_size = 0;
static int sum_sizes(const char *pathname, const struct stat *statbuf, int typeflag, struct FTW *ftwbuf)
{
/*if stat failed on the current file*/
if(typeflag == FTW_NS)
{
printf("No stats (permissions ?) on %s", pathname);
return 0;
}
total_size = total_size + (long long int ) statbuf->st_size;
total_real_size = total_real_size + (long long int ) ( statbuf->st_blocks * 512 );
return 0;
}
最後に、累積サイズを表示します。
printf("total size: %lld (%0.2lf K %0.2lf M)\n", total_size, (float)total_size /1024.0, (float)total_size /(1024.0*1024.0));
printf("total real size: %lld (%0.2lf K %0.2lf M)\n", total_real_size, (float)total_real_size /1024.0, (float)total_real_size /(1024.0*1024.0));
値を du と比較すると、いくつかの違いがあります
time ./scan_dir ~/
====>
total size: 15208413390 (14851966.00 K 14503.87 M)
total real size: 15708553216 (15340384.00 K 14980.84 M)
block size : 4096 / fond. block size : 4096
fs size: 22.7895 G
./scan_dir ~/ 0,03s user 0,24s system 98% cpu 0,277 total
time du -s ~/
15119876 /home/cedlemo/
du -s ~/ 0,07s user 0,22s system 98% cpu 0,287 total
注: du のマニュアル ページを読んだ後、du は私の小さなアプリケーション scan_dir とほぼ同じ動作をすることがわかりました (マウント ポイントをスキップし、シンボリック リンクを参照解除し、Ko の値を計算するために 1024 を使用します)。
私のアプリケーションで見つかったより近い値は、実際のサイズ (使用されているブロック) の合計のようですが、値は実際には同じではありません。
この違いの理由は何ですか? 私が間違っていることは何ですか?