QFileSystemWatcher 不在控制台应用程序中发出 fileChanged()

QFileSystemWatcher does not emit fileChanged() in console application

在我的文件 console.h/.cpp 中,我有一个小的 class,它只要求用户输入一些文本,然后打印文本,直到用户输入 "quit"(参见方法 consoleMain())。但是,在 main.cpp 我也有一个 QFileSystemWatcher 监视文件 MyTextFile.txt 并在文本文件更改时调用 Console::slotFileChanged(QString)。不幸的是 QFileSystemWatcher 不起作用。 Console::slotFileChanged(QString) 在我更改文本文件时从不执行。据我所知, QFileSystemWatcher 仅在主事件循环已启动时才有效,我的代码也是如此。 当我禁用 main.cpp 中的 QTimer::singlaShot 并将其替换为 emit console.signalStart() 主事件循环不会进入,但是我在进入"quit"后看到QFileSystemWatcher("File changed!")的消息。 问题是:是否可以让用户与控制台交互并让 FileWatcher 在并行更改文本文件时发出信号? (我也尝试将 QFileSystemWatcher 放入控制台 class 并在堆上创建它;不幸的是它没有改变任何东西)

这是我的代码:

console.h

#ifndef CONSOLE_H
#define CONSOLE_H

#include <iostream>
#include <QObject>
#include <QFileSystemWatcher>

class Console: public QObject
{
    Q_OBJECT

public:

    Console(QObject *parent = 0);
    ~Console();

signals:

    void signalStart();
    void signalEnd();

public slots:

    void consoleMain();
    void slotFileChanged(QString text);
    void slotEmit();
};

#endif // CONSOLE_H

console.cpp

#include "console.h"

Console::Console(QObject *parent): QObject(parent)
{
}

Console::~Console()
{
}

void Console::consoleMain()
{
    bool isRunning = true;
    std::string in;

    while (isRunning)
    {
        std::cout << ">" << std::flush;
        std::getline(std::cin, in);

        if (in.compare("quit") == 0)
            isRunning = false;
        else
            std::cout << "You have entered: " << in << std::endl;
    }

    emit signalEnd();
}

void Console::slotFileChanged(QString text)
{
    Q_UNUSED(text);
    std::cout << "File changed!" << std::endl;
}  

void Console::slotEmit()
{
    emit signalStart();
}

main.cpp

#include "console.h"
#include <QCoreApplication>
#include <QTimer>

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

    QFileSystemWatcher watcher(&a);
    watcher.addPath("C:/MyTextFile.txt");

    Console console(&a);

    QObject::connect(&console, SIGNAL(signalStart()), &console, SLOT(consoleMain()));
    QObject::connect(&console, SIGNAL(signalEnd()), &a, SLOT(quit()));
    QObject::connect(&watcher, SIGNAL(fileChanged(QString)), &console, SLOT(slotFileChanged(QString)));

    QTimer::singleShot(0, &console, SLOT(slotEmit()));
    //emit console.signalStart();

    std::cout << "Enter main event loop now" << std::endl;
    return a.exec();
}

好的,解决了。我已经使用不同的线程尝试了 Yakk 的想法(感谢 Yakk 的想法)。我不得不引入一个名为 MyObjectQObject 的新子类。在其构造函数中,我为控制台对象创建了 Console 和一个新的 QThreadQFileSystemWatcher 是在 main.cpp 中创建的,也是 MyObjcet 的一个实例。请参阅下面的代码:

myobject.h

#ifndef MYOBJECT_H
#define MYOBJECT_H

#include "console.h"
#include <QThread>
#include <QCoreApplication>

class MyObject : public QObject
{
    Q_OBJECT

public:

    MyObject(QObject *parent = 0);
    ~MyObject();

private:

    QThread *thread;
    Console *console;

signals:

    void signalStart();

public slots:

    void slotFileChanged(QString text);
    void slotEnd();
};

#endif // MYOBJECT_H

myobject.cpp

#include "myobject.h"

MyObject::MyObject(QObject *parent): QObject(parent)
{
    console = new Console;

    thread = new QThread(this);
    console->moveToThread(thread);
    thread->start();

    connect(this, SIGNAL(signalStart()), console, SLOT(consoleMain()));
    connect(console, SIGNAL(signalEnd()), this, SLOT(slotEnd()));
    emit signalStart();
}

MyObject::~MyObject()
{
    thread->quit();
    thread->wait();
}

void MyObject::slotFileChanged(QString text)
{
    console->displayChangedFileText(text);
}

void MyObject::slotEnd()
{
    QCoreApplication::exit(0);
}

main.cpp

#include "myobject.h"
#include <QTimer>

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

    QFileSystemWatcher *watcher = new QFileSystemWatcher(&a);
    watcher->addPath("C:/MyTextFile.txt");

    MyObject *object = new MyObject(&a);

    QObject::connect(watcher, SIGNAL(fileChanged(QString)), object, SLOT(slotFileChanged(QString)));

    std::cout << "Enter main event loop now" << std::endl;
    return a.exec();
}

console.h/.cpp未改,只有Console::slotFileChanged(QString)Console::displayChangedFileText(QString).

代替