Добавление одного файла в другой

В примере кода в этом разделе показано, как открывать и закрывать файлы, читать и записывать в файлы, а также блокировать и разблокировать файлы.

В этом примере приложение добавляет один файл в конец другого файла. Сначала приложение открывает добавляемый файл с разрешениями, которые позволяют только приложению выполнять запись в него. Однако во время процесса добавления другие процессы могут открывать файл с разрешением только для чтения, что обеспечивает snapshot представление добавляемого файла. Затем файл блокируется во время фактического процесса добавления, чтобы обеспечить целостность данных, записываемых в файл.

В этом примере не используются транзакции. Если бы вы использовали транзакции, у вас был бы доступ только для чтения. В этом случае вы увидите добавленные данные только после завершения операции фиксации транзакции.

В примере также показано, что приложение открывает два файла с помощью CreateFile:

  • One.txt открывается для чтения.
  • Two.txt открыт для записи и совместного чтения.

Затем приложение использует ReadFile и WriteFile для добавления содержимого One.txt в конец Two.txt путем чтения и записи блоков размером 4 КБ. Однако перед записью во второй файл приложение использует SetFilePointer для установки указателя второго файла на конец этого файла, а lockFile — для блокировки области записи. Это не позволит другому потоку или процессу с повторяющимся дескриптором получить доступ к области во время выполнения операции записи. После завершения каждой операции записи unlockFile используется для разблокировки заблокированной области.

#include <windows.h>
#include <stdio.h>

void main()
{
  HANDLE hFile;
  HANDLE hAppend;
  DWORD  dwBytesRead, dwBytesWritten, dwPos;
  BYTE   buff[4096];

  // Open the existing file.

  hFile = CreateFile(TEXT("one.txt"), // open One.txt
            GENERIC_READ,             // open for reading
            0,                        // do not share
            NULL,                     // no security
            OPEN_EXISTING,            // existing file only
            FILE_ATTRIBUTE_NORMAL,    // normal file
            NULL);                    // no attr. template

  if (hFile == INVALID_HANDLE_VALUE)
  {
     printf("Could not open one.txt."); 
     return;
  }

  // Open the existing file, or if the file does not exist,
  // create a new file.

  hAppend = CreateFile(TEXT("two.txt"), // open Two.txt
              FILE_APPEND_DATA | FILE_GENERIC_READ,    // open for appending and locking
              FILE_SHARE_READ,                         // allow multiple readers
              NULL,                                    // no security
              OPEN_ALWAYS,                             // open or create
              FILE_ATTRIBUTE_NORMAL,                   // normal file
              NULL);                                   // no attr. template

  if (hAppend == INVALID_HANDLE_VALUE)
  {
     printf("Could not open two.txt."); 
     return;
  }

  // Append the first file to the end of the second file.
  // Lock the second file to prevent another process from
  // accessing it while writing to it. Unlock the
  // file when writing is complete.

  while (ReadFile(hFile, buff, sizeof(buff), &dwBytesRead, NULL)
      && dwBytesRead > 0)
    {
    dwPos = SetFilePointer(hAppend, 0, NULL, FILE_END);
    if (!LockFile(hAppend, dwPos, 0, dwBytesRead, 0))
    {
       printf("Could not lock two.txt");
    }
    WriteFile(hAppend, buff, dwBytesRead, &dwBytesWritten, NULL);
    UnlockFile(hAppend, dwPos, 0, dwBytesRead, 0);
    }

  // Close both files.

  CloseHandle(hFile);
  CloseHandle(hAppend);
}