Подтвердить что ты не робот

Как я могу сделать атомную запись/добавление в С# или как получить файлы, открытые с помощью флага FILE_APPEND_DATA?

В большинстве операционных систем Unix и Posix, выполняющих открытый() операционный системный вызов с 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

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

4b9b3361

Ответ 1

Используйте одну из перегрузок конструктора 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);
            }
        }
    }

Ответ 2

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

Ответ 3

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

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

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