CHtmlView::Navigate2 和锁定的文件

CHtmlView::Navigate2 and locked files

我发现,如果我导航到一个文件,然后尝试删除该文件,系统会告诉我它正在使用中。

我怎样才能阻止文件成为 "in use",这样我就可以删除它并重新创建它以更新 html 显示?

如果我每次都创建一个新的 XML 数据文件并导航到该文件,那么我不会遇到任何问题。这是因为没有要删除的文件。

但是当我使用同一个文件时,我发现文件正在使用中。

我在对话框 OnDestroy 方法中添加了代码,并添加了我创建的临时文件数组。然后我尝试删除它们:

for (auto i = 0; i < m_aryStrTempFiles.GetCount(); i++)
{
    if (PathFileExists(m_aryStrTempFiles[i]))
    {
        if (!::DeleteFile(m_aryStrTempFiles[i]))
        {
            AfxMessageBox(theApp.GetLastErrorAsString(), MB_OK | MB_ICONERROR);
        }
    }
}

我发现所有文件都被认为仍在使用中。

创建临时文件名的代码不是问题所在:

CString CMeetingScheduleAssistantApp::GetFolderTempFilenameEx(CString strFolder, CString strToken, CString strSuffix /*_T("htm")*/)
{
    CString     strFile;
    int         i;

    ::GetTempFileName(strFolder, strToken, 0, strFile.GetBuffer(_MAX_PATH));
    strFile.ReleaseBuffer();

    // Because we will rename to .HTM we must delete old file
    ::DeleteFile(strFile);

    // I can't be sure the suffix is .tmp so I manually
    // replace the suffix, whatever it is, with .htm"
    i = strFile.ReverseFind(_T('.'));
    strFile = strFile.Left(i + 1);
    strFile += strSuffix;

    return strFile;
}

这是保存我的 XML 文件的代码:

bool CMeetingScheduleAssistantApp::SaveToXML(CString strFileXML, tinyxml2::XMLDocument& rDocXML)
{
    FILE    *fStream = nullptr;
    CString strError, strErrorCode;
    errno_t eResult;
    bool    bDisplayError = false;
    int     iErrorNo = -1;

    using namespace tinyxml2;

    // Does the file already exist?
    if (PathFileExists(strFileXML))
    {
        // It does, so try to delete it
        if (!::DeleteFile(strFileXML))
        {
            // Unable to delete!
            AfxMessageBox(theApp.GetLastErrorAsString(), MB_OK | MB_ICONINFORMATION);
            return false;
        }
    }

    // Now try to create a FILE buffer (allows UNICODE filenames)
    eResult = _tfopen_s(&fStream, strFileXML, _T("w"));
    if (eResult != 0 || fStream == nullptr) // Error
    {
        bDisplayError = true;
        _tcserror_s(strErrorCode.GetBufferSetLength(_MAX_PATH), _MAX_PATH, errno);
        strErrorCode.ReleaseBuffer();
    }
    else // Success
    {
        // Now try to save the XML file
        XMLError eXML = rDocXML.SaveFile(fStream);
        int fileCloseResult = fclose(fStream);
        if (eXML != XMLError::XML_SUCCESS)
        {
            // Error saving
            bDisplayError = true;
            strErrorCode = rDocXML.ErrorName();
            iErrorNo = rDocXML.GetErrorLineNum();
        }

        if (!bDisplayError)
        {
            if (fileCloseResult != 0)
            {
                // There was a problem closing the stream. We should tell the user
                bDisplayError = true;
                _tcserror_s(strErrorCode.GetBufferSetLength(_MAX_PATH), _MAX_PATH, errno);
                strErrorCode.ReleaseBuffer();
            }
        }
    }

    if (bDisplayError)
    {
        if (iErrorNo == -1)
            iErrorNo = errno;

        strError.Format(IDS_TPL_ERROR_SAVE_XML, strFileXML, strErrorCode, iErrorNo);
        AfxMessageBox(strError, MB_OK | MB_ICONINFORMATION);

        return false;
    }

    return true;
}

如你所见,他们都关闭了流。然而,即使在 OnDestroy 中我先删除了 html 视图,临时文件仍然无法删除。为什么?

问题是我如何测试仍在打开的文件:

bool CMeetingScheduleAssistantApp::WaitForFileToBeReady(CString strFile)
{
    HANDLE hFile;
    int delay = 10;

    while ((hFile = CreateFile(strFile, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL)) == INVALID_HANDLE_VALUE)
    {
        if (GetLastError() == ERROR_SHARING_VIOLATION) {
            Sleep(delay);
            if (delay < 5120) // max delay approx 5.Sec
                delay *= 2;
        }
        else
        {
            AfxMessageBox(theApp.GetLastErrorAsString(), MB_OK | MB_ICONINFORMATION);
            return false; // some other error occurred
        }
    }

    if (hFile != INVALID_HANDLE_VALUE)
        CloseHandle(hFile);

    return true;
}

我遗漏了 CloseHandle 行代码。