之前有过对Qt的QFile以Text纯文本方式进行读取时的学习,这两天由于实时需要又对QFileSystemWatcher(这个类是干什么用的)进行了学习,发现也是问题很让人头疼.
我想监视一个文件夹(比如:"F:QtWorkSpace")里面某些特定文件(如当天以时间戳命名的日志文件)的监视,这个文件夹里面的文件不会被删除,只会有新的文件添加,而这些文件也只会被增加内容,那么假设我在我的QObject对象实例中定义的监视器QFileSystemWatcher命名为m_watcher,我在构造函数中的connect语句如下:
1
2
3
4
|
connect(m_watcher,SIGNAL(directoryChanged( const QString)), this ,SLOT(on_DirectoryChanged(QString))); connect(m_watcher,SIGNAL(fileChanged(QString)), this , SLOT(on_FileChanged( const QString))); |
(on_DirectoryChanged和on_FileChangeed都是自定义的slot)首先,比如说我早上八点起床,开启我的应用程序之后,我监视的文件就是以这个时刻的时间戳命名的日志文件(加入命名规则为<时间戳+".log":hh_mm.log>),那么这个文件名称就是"08_00.log",在构造函数里面直接m_watcher->addPath("08_00.log").过了一会有后台程序往里面写日志,那么在on_FileChanged函数里面我的处理是,直接把被监视的文件中所添加的内容显示在程序的界面上,这个没有问题.而我在on_DirectoryChanged函数里面是这样的:首先,获取当前的时间戳,然后通过之前文件命名规则的到一个QString newLog,我先对比这个newLog和我之前正在监视的日志文件名称,如果一样的话,我就什么也不做,如果不一样,我就把之前的监视的文件从监视器里面去除掉(通过m_watcher的removePath),然后把newLog添加到监视器里面去进行监视,之前的那个文件就不管了.
于是我就想当然的这么做了,但是最后的结果却出乎意料.
如果我在on_DirectoryChanged打上了断点进行调试,只要我按规矩添加新的日志文件,然后往这个文件里添加新的内容,保存之后,界面上就会如我所预期的那样把新添加的内容给显示出来.然后呢,我就把断电删除点,进行同样的文件添加和操作,却发现什么也没有显示出来,我往新文件里添加的东西并没有如我所愿的被显示出来.而查看"应用程序输出"的时候发现,里面会输出"添加hh_mm.log文件失败.".
后来查看QFileSystemWatcher的CPP文件源代码http://www.oschina.net/code/explore/qt-4.7.1/src/corelib/io/qfilesystemwatcher.cpp的时候才发现,这个类的实现有一个临界区,于是恍然大悟.因为signal和slot这个机制不是异步的,如果我都在on_DirectoryChanged的函数体里面进行removePath和addPath两个动作的时候,其中必然会有一个失败而另一个成功(不考虑文件路劲不存在的情况或者是路径已经被remove或者add的情况),所以这两个动作只能有一个在这个函数里面执行.
所以我最后的解决办法是,只在on_DirectoryChanged里面addPath,而在on_FileChanged里面removePath,也就是等待m_watcher对临界区解锁之后进行remove就行了.
http://www.oschina.net/question/658193_118503