QFlags 枚举类型转换突然失败

QFlags Enum Type Conversion fails all of a sudden

本文关键字:失败 突然 类型转换 枚举 QFlags      更新时间:2023-10-16

我已经在库中运行了很长时间:

MyClass::MyClass() 
  : QDialog()
{
    // (...)
    setWindowFlags( Qt::CustomizeWindowHint | Qt::WindowTitleHint );
    // (...)
}

然后,在更改了库的各个部分后,我突然收到以下消息:

error C2664: 'QWidget::setWindowFlags': cannot convert parameter 1 from 'int' to 'Qt::WindowFlags'

显然,它找不到 QFlags 类提供的 | 运算符重载,因此 | 的结果返回 int 而不是 QFlags 构造。

我知道我可以手动将结果转换为(Qt::WindowFlags)并使其工作,但 QFlags 通常会使这种强制转换变得不必要。

知道什么样的变化会导致这种行为吗?

我包括<QtGui/QDialog>通常就足够了。包括<QtCore/QFlags>不会改变行为。

从 5.12.0 开始,这应该通过以下提交来修复:"声明运算符|在该命名空间中的 QFlags 的 Qt 命名空间中"。在 5.12.0 之前,Qt 习惯于将其枚举运算符放在全局命名空间中(参见 qnamespace.h),而不是放在 Qt 命名空间中。

问题是,当当前命名空间中存在另一个匹配的运算符时,编译器将不会搜索父范围。因此,只要您为命名空间中的任何类型添加运算符的重载,Qt 的重载就不会在匹配集中。ADL 通常用于解析在与类型相同的命名空间中声明的运算符,但如果运算符位于不同的命名空间中,则此操作不起作用。

真正的解决方案是让Qt将运算符放在与他们操作的类型相同的命名空间中,他们在5.12.0中就是这样做的。如果您坚持使用以前的版本,您可以自己导入运算符:

using ::operator|;
setWindowFlags(Qt::CustomizeWindowHint | Qt::WindowTitleHint);

请注意,这可能会产生意想不到的后果,因为您可能会在不应该使用的上下文中提供大量名称。

检查源代码,看看在出现编译错误之前,是否使用 Q_DECLARE_OPERATORS_FOR_FLAGS 在命名空间中的某个地方声明自己的标志。

正如@isanae所说,If there is another operator that matches in the current namespace, ....因此,您不应放置可能匹配的其他运算符。

解决方案是将所有自己的Q_DECLARE_OPERATORS_FOR_FLAGS声明放在全局命名空间中,就像Qt所做的那样。

希望这会有所帮助。当我将所有内容放入 unity 构建环境中时,我遇到了同样的问题,我切换了源代码的顺序,最终找到了导致此问题的代码片段:一个Q_DECLARE_OPERATORS_FOR_FLAGS命名空间内。

您是否尝试过将 | 表达式与函数调用分开?像这样:

// ..
Qt::WindowFlags flags = Qt::CustomizeWindowHint | Qt::WindowTitleHint;
setWindowFlags( flags );
// ...

只是为了看看问题到底在哪里...

如果是包含的问题,只需#include <QtGui>