Как сделать атомарную запись / добавление в C # или как открыть файлы с флагом FILE_APPEND_DATA? - PullRequest
21 голосов
/ 07 декабря 2009

В большинстве операционных систем, соответствующих Unix и Posix, выполнение вызова операционной системы open () с O_APPEND указывает ОС, что запись должна быть атомарной операцией добавления и записи. При таком поведении для локальных файловых систем, когда вы делаете запись, вы знаете, что она добавляется в конец файла.

Операционные системы Windows поддерживают ту же функциональность, передав FILE_APPEND_DATA в соответствующем параметре системному вызову Win32 CreateFile ().

ссылки:

http://www.google.com/search?q=msdn+createfile
or: http://msdn.microsoft.com/en-us/library/aa363858(VS.85).aspx

http://www.google.com/search?q=msdn+IoCreateFileSpecifyDeviceObjectHint
or: http://www.google.com/search?q=msdn+IoCreateFileSpecifyDeviceObjectHint

Моя проблема в том, что я не могу определить, как получить это поведение в C #, используя Net Framework Библиотеки, есть ли способ получить такое поведение с помощью Net Framework? Я не верю, что использование FileMode.Append, кстати, дает такое поведение.

Ответы [ 3 ]

34 голосов
/ 31 марта 2011

Используйте одну из перегрузок конструктора FileStream:

new FileStream(FileName, FileMode.Open, FileSystemRights.AppendData,
            FileShare.Write, 4096, FileOptions.None)

FileSystemRights.AppendData соответствует FILE_APPEND_DATA

FileStream, похоже, настаивает на буферизации, поэтому убедитесь, что буфер достаточно велик для каждого пишите и звоните Flush() после каждой записи.

Крошечный пример:

    private void button1_Click(object sender, EventArgs e) {
        Thread t1 = new Thread(DoIt);
        Thread t2 = new Thread(DoIt);
        t1.Start("a");
        t2.Start("b");
        Thread.Sleep(2000);
        Environment.Exit(0);
    }

    private void DoIt(object p) {
        using (FileStream fs = new FileStream(FileName, FileMode.Open, FileSystemRights.AppendData,
            FileShare.Write, 4096, FileOptions.None)) {
            using (StreamWriter writer = new StreamWriter(fs)) {
                writer.AutoFlush = true;
                for (int i = 0; i < 20; ++i)
                    writer.WriteLine("{0}: {1:D3} {2:o} hello", p, i, DateTime.Now);
            }
        }
    }
1 голос
/ 29 марта 2011

Вы можете вызвать CreateFile , используя PInvoke с необходимыми параметрами, и передать полученный дескриптор одному из Конструкторов FileStream , который принимает SafeFileHandle в качестве параметра.

0 голосов
/ 08 декабря 2009

Почему вы не можете использовать

System.IO.File.AppendAllText("C:\\somefile.txt", "some content");

? Это также потокобезопасный / "атомарный" вызов.

...