CHtmlView :: Navigate2 и заблокированные файлы - PullRequest
0 голосов
/ 05 апреля 2019

Я обнаружил, что если я перехожу к файлу, а затем пытаюсь удалить файл, система сообщает мне, что он используется.

Как я могу остановить использование файлов, чтобы я мог удалить их и создать заново для обновления 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-представление, временные файлы все еще не могут быть удалены. Почему?

1 Ответ

0 голосов
/ 05 апреля 2019

Проблема заключалась в том, как я проверял файл, который все еще открыт:

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 строк кода.

...