1

Linux環境でCコードを開発しています。私はfwriteいくつかのデータをいくつかのファイルに書き込んでいます。プログラムは、頻繁に (少なくとも 1 日に 1 回) 停電が発生する環境で実行されます。fwriteしたがって、データの書き込み中に停電が発生した場合、ファイルが更新されないようにしたいと考えています。fwriteがジョブを終了したときにのみファイルを保存する必要があります。fwrite書き込みプロセスを終了するだけでファイルに影響を与えることをどのように使用できますか?

編集: fopen を使用wbして、ファイル内の以前の情報を破棄し、新しいファイルを書き込みます。

FILE *rtng_p;
rtng_p = fopen("/etc/routing_table", "wb");
fwrite(&user_list, sizeof(struct routing), 40, rtng_p);

数バイトの長さの非常に小さなデータです

4

3 に答える 3

6

最初に、ファイルを同じファイル システムの一時パスに書き込みます/etc/routing_table.tmp。次に、元のファイルの上にコピーの名前を変更します。名前の変更はアトミックであることが保証されています。

したがって、呼び出しのシーケンスは、、、、、fopenfwriteなりfcloseますrename

于 2013-06-25T18:22:44.443 に答える
2

David Schwartz answerに示されているシーケンスに加えて、たとえばflock(2) syscall (またはlockf(3) ie fcntl(2) with F_SETLK....)などでアドバイザリ ロックを使用することもできます。

それは、直後に追加することを意味します

 FILE * fil = fopen("/etc/routing_table.tmp", "wb");

台詞

 if (!fil) 
   { perror("/etc/routing_table.tmp"); exit(EXIT_FAILURE); };
 if (flock(fileno(fil), LOCK_EX)) 
   { perror("flock LOCK_EX"); exit(EXIT_FAILURE); };

そして最後に、あなたは

 if (fflush(fil)) /* flush the file before unlocking it!!*/
   { perror("fflush"); exit(EXIT_FAILURE); };
 if (flock(fileno(fil), LOCK_UN))
   { perror("flock LOCK_UN"); exit(EXIT_FAILURE); };
 if (fclose (fil))
   { perror("fclose"); exit(EXIT_FAILURE); };;
 if (rename("/etc/routing_table.tmp", "/etc/routing_table"))
   { perror("rename"); exit(EXIT_FAILURE); };

このようなアドバイザリ ロックを使用すると、プログラムの 2 つのプロセスが実行されている場合でも、ファイルを書き込むのは 1 つだけになります。

しかし、それはおそらくやり過ぎです。

ところで、あなたはバイナリデータを に書き込んでいるようです/etc/。私はそれが習慣や慣習に反していると信じています ( Linux Filesystem HierarchyまたはLinux Standard Baseを参照)。/etc以下のファイルはテキストであることを期待しています。おそらく、ファイルを の下に置きたいです/var/libか?

Advanced Linux Programming book onlineも参照してください。

于 2013-06-25T19:33:49.477 に答える