先看一个例子代码,首先是项目文件qhash.pro,只有一行。
SOURCES += main.cpp
然后是程序文件main.cpp,
#include <QList>
#include <QPointF>
class Q_DECL_EXPORT PointList : public QList<QPointF>
{
};
int main(int argc, char *argv[])
{
return 0;
}
在Qt5.10.1 + V2015的环境下编译,报错:
error: C2665: “qHash”: 25 个重载中没有一个可以转换所有参数类型 ....
首先一点,这个错误没有问题,qt的源码中确实没有实现关于QPointF的qHash函数。
但是诡异的是,因为刚从Qt4迁移到Qt5的环境中,而同样的程序在Qt4.8.4的环境中是没有错误的,只有一堆警告,
warning: C4661: “QSet<T> QList<T>::toSet(void) const”: 没有为显式模板实例化请求提供适当的定义
with
[ T=QPointF ]
注: 这里的意思是QList中的toSet函数,返回QSet类型的对象,但实例化QSet时,没有找到QSet的定义。但是因为实际没有使用toSet这个函数,所以只是给出了警告信息。
这是什么原因呢?个人觉得非常奇怪。通过查找Qt的源码发现,在Qt5.10的include目录下,包含了两个QList文件,一个在QtCore目录下,另一个在QtGui的目录下,因为我这个项目文件中没有 Qt -= gui,所以默认是包含了Gui模块,这里的#include指令找到的是QtGui目录下的QList文件,怪异的是,这个文件的内容如下:
#include "qevent.h"
不符合Qt头文件的惯例,而QtCore模块下的文件就是合理的。
#include "qlist.h"
这里猜测是QtGui中的QList文件内容错误,或者这个文件名根本不应该是QList,而应该改名为QEvent。这应该是Qt的一个BUG,在Qt的bug系统中也有相关的问题。
经过验证,把Qt目录下的include/QtGui/QList文件改名未QEvent,上面的程序编译没有错误,和4.8环境一样给出了一组警告信息。

在Qt5.10.1 + V2015环境下,一个使用QList的项目报错,由于缺少QPointF的qHash函数实现。与Qt4.8.4相比,虽然同样报警告,但在Qt5中导致编译错误。问题源自QtGui目录下的QList头文件内容异常,可能为Qt的BUG。将QtGui/QList重命名为QEvent后,程序能正常编译并仅显示警告。

被折叠的 条评论
为什么被折叠?



