Evgenii Legotckoi
Мамыр 11, 2020, 3:38 Т.Қ.

Qt WinAPI - Сабақ 010. Басқа процесс файлды жазуға рұқсат етілмегенін қалай тексеруге болады.

Бір қызығы, кәсіби мансабымның барысында мен QFile-дің бір қызықты мүмкіндігіне тап болдым. Оның көмегімен файлға ақпаратты оқу немесе жазу мүмкіндігін тексеруге болады. Бірақ сонымен бірге, егер файл бір бағдарламаның бірнеше даналары арқылы ашылған болса, QFile файлдың оқу және жазу рұқсаттарын елемейді. Бұл дегеніміз, егер файл бағдарламаның басқа данасында ашылса, QFile бұл файлды оқылатын немесе жазылатын ретінде анықтайды, сонымен қатар файлға сәтті жаза алады.


Мәселенің сипаттамасы

Жазуға тыйым салуды тексеру келесідей орындалуы мүмкін, бірақ бұл әрқашан жұмыс істемейді, мен жоғарыда айттым. Бірақ сіз оны осылай тексере аласыз

bool isFileWritable(const QString& fileName)
{
     QFile file(fileName);
     bool isWritable = file.open(QFile::ReadWrite);
     file.close();
     return isWritable;
}

Дегенмен, мақаланың ең басындағы мәлімдемені келесі кодты пайдаланып тексере аласыз.

#include <QCoreApplication>

#include <QFile>
#include <QDebug>
#include <QDateTime>

int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);

    // Подставьте любой удобный для вас путь к файлу
    QFile file("D:/check.txt");
    file.write(QDateTime::currentDateTime().toString().toLatin1());
    file.flush();

    return a.exec();
}

Бағдарламаны құрастырғаннан кейін бірнеше дананы бірге іске қосыңыз және файл мазмұны соңғы іске қосу данасы уақытына сәйкес келетінін көресіз. Яғни, QFile файлдың басқа процестердің рұқсаттары мен иелігін толығымен елемейді. Дегенмен, егер сіз осы бағдарламаны іске қосып, содан кейін стандартты блокнот сияқты басқа бағдарламамен файлдың мазмұнын қайта жазуға әрекеттенсеңіз, келесі хабарландыруды аласыз.

Жауап

Мен бұл жағдайдан шығудың жолын WinAPI-де платформаға тән функционалдылықты, мүмкін басқа операциялық жүйелер үшін аналогтарды пайдалану арқылы таптым.

Ол үшін файлды иеленетін std::wofstream кіріс/шығыс ағынына бірегей көрсеткіш жасау керек, оны мысалы, MyFileBlocker класына қоюға болады.
Файл блокаторына арналған орынды ұйымдастыру мәселесі сіздің қалауыңызда қалады.

std::unique_ptr<std::wofstream> m_openedFile;

Содан кейін файлды құлыптау және құлпын ашу үшін екі функцияны жазыңыз.

void MyFileBlocker::unlockFile()
{
    m_openedFile.reset(nullptr);
}

void MyFileBlocker::relockFile(std::wstring fileName)
{
    m_openedFile.reset(new std::wofstream());
    // _SH_DENYWR is WinAPI dependent deny write mode
    m_openedFile->open(fileName, std::ios_base::app, _SH_DENYWR);
    if (!m_openedFile->is_open())
    {
        m_openedFile.reset(nullptr);
    }
}

Содан кейін сіз қазірдің өзінде тексере аласыз, егер көрсеткіш бар болса, файлға жазу әрекеттерін орындауға болады.

if (m_openedFile)
{
    // You can write to file
}

Сондай-ақ, бұл мәселені бағдарламаның іске қосылған процесін біріктіру арқылы шешуге болады, осылайша әрбір дананы файл рұқсаттары тұрғысынан бірегей деп анықтауға болады.
Бірақ шынымды айтсам, бұл мәселені шешу кезінде бұл менің ойыма келмеді, сондықтан мен бұл мәселені std::wofstream арқылы шештім.

Осы тақырып бойынша ұсынылатын мақалалар

Мақала бойынша сұралады0сұрақтар(лар)

1

Ол саған ұнайды ма? Әлеуметтік желілерде бөлісіңіз!

Пікірлер

Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
Кіріңіз немесе Тіркеліңіз