先看一个例子代码,首先是项目文件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环境一样给出了一组警告信息。