Могу ли я использовать CreateFile, но заставить дескриптор в std :: ofstream? - PullRequest
17 голосов
/ 24 января 2009

Есть ли способ воспользоваться преимуществами флагов создания файлов в Win32 API, таких как FILE_FLAG_DELETE_ON_CLOSE или FILE_FLAG_WRITE_THROUGH, как описано здесь http://msdn.microsoft.com/en-us/library/aa363858(VS.85).aspx, но затем принудительно использовать этот дескриптор в std :: ofstream?

Интерфейс ofstream явно не зависит от платформы; Я хотел бы принудительно установить некоторые зависимые от платформы настройки в «под капотом».

Ответы [ 2 ]

25 голосов
/ 24 января 2009

Возможно прикрепить C ++ std::ofstream к дескриптору файла Windows. В VS2008 работает следующий код:

HANDLE file_handle = CreateFile(
    file_name, GENERIC_WRITE,
    0, NULL, CREATE_ALWAYS,
    FILE_ATTRIBUTE_NORMAL, NULL);

if (file_handle != INVALID_HANDLE_VALUE) {
    int file_descriptor = _open_osfhandle((intptr_t)file_handle, 0);

    if (file_descriptor != -1) {
        FILE* file = _fdopen(file_descriptor, "w");

        if (file != NULL) {
            std::ofstream stream(file);

            stream << "Hello World\n";

            // Closes stream, file, file_descriptor, and file_handle.
            stream.close();

            file = NULL;
            file_descriptor = -1;
            file_handle = INVALID_HANDLE_VALUE;
        }
}

Это работает с FILE_FLAG_DELETE_ON_CLOSE, но FILE_FLAG_WRITE_THROUGH может не дать желаемого эффекта, так как данные будут буферизироваться объектом std::ofstream и не будут записываться непосредственно на диск. Однако все данные в буфере будут сброшены в ОС при вызове stream.close().

1 голос
/ 23 июня 2010

Некоторые из этих флагов также доступны при использовании _fsopen / fopen:

FILE* pLockFile = _fsopen(tmpfilename.c_str(), "w", _SH_DENYWR );
if (pLockFile!=NULL
{
   // Write lock aquired
   ofstream fs(pLockFile);
}

Здесь мы открываем файл, поэтому при выполнении сброса он записывается (и удаляется при закрытии):

FILE* pCommitFile = fopen(tmpfilename.c_str(), "wcD");
if (pCommitFile!=NULL)
{
   // Commits when doing flush
   ofstream fs(pCommitFile);
}
...