QT之QFlags详解

简介: QT之QFlags详解

学习之前我们先补习一下枚举和或运算的基础,因为很多人直接学Qt的或者c++忘的有点多的,需要有一些基础才能搞懂QFlag这个东西。

枚举:

枚举类型(Enum Types)是一种用户定义的数据类型,用于表示具名的整数常量。枚举类型可以帮助提高代码的可读性,使程序更易于理解。

以下是一些使用枚举类型的典型情况:

代替魔法数值: 枚举类型可以用于替代代码中的魔法数值(Magic Numbers)。例如,假设你的程序中有一个表示星期的整数,你可以使用枚举类型:
enum Weekday {
    Monday,
    Tuesday,
    Wednesday,
    Thursday,
    Friday,
    Saturday,
    Sunday
};

这样你的代码中就可以使用 Weekday 枚举来表示星期,而不是使用 0 到 6 之间的数字。

有限的选项集: 当你的变量只能有有限的一组选项时,使用枚举类型可以提高代码的清晰度。例如,表示颜色的枚举:
enum Color {
    Red,
    Green,
    Blue,
    Yellow,
    // ...
};

这样你可以使用 Color 枚举来表示颜色,而不是使用数字或字符串。

在以上代码中这个Weekday和Color是枚举类型,其中Red或者Sunday是枚举常量

枚举类型与运算

枚举类型的或运算通常用于创建一个组合值,其中一个枚举值的位被设置。这在处理一组选项或标志时非常有用。以下是一个简单的例子:

假设有一个表示文件权限的枚举类型:

enum FileAccess { Read = 1, Write = 2, Execute = 4 };

现在,假设你有一个文件,你想给它读和写的权限:

FileAccess myFilePermissions = Read | Write;

在这里,| 是按位或运算符,用于将 ReadWrite 枚举值的对应位进行组合。现在,myFilePermissions 的值是 Read | Write 的组合,它的二进制表示为11;所以值就是十进制的3

计算过程

关键词复习:组合值(类型枚举的组合)(或运算)这种逻辑运算有时候也叫做位掩码(bitmasks)

进入正题QFlags:

首先说一下普通枚举得缺点:

QFlags<Enum>类是一个模板类,其中Enum是枚举类型。QFlags在Qt中用于存储枚举值的组合。

用于存储或组合枚举值的传统C++方法是使用整型变量。这种方法的不便之处在于根本没有类型检查,任何枚举值都可以与任何其他枚举值进行逻辑运算。

enum Orientation
    {
        Up = 1,
        Down = 2,
        Left = 4,
        Right = 8,
    };
    enum Direction
    {
        horizontal = 2,
        vertical = 3,
    };

这两种操作编译器不会报错:

Orientation::Up | Direction::horizontal;
Orientation::Up | Orientation::Down;

第一种两个不相关的枚举值做逻辑运算没有意义,第二种运算结果是3,但Orientation中没有值是3的标识符。

在Qt6中c++开发指南中是这样介绍QFlags的:

QFlags<Enum>是一个模板类,其中Enum是枚举类型,QFlags用于定义枚举值的或运算组合,在Qt中经常用到 QFlags 类。例如,QLabel 有一个alignment 属性,其读写函数分别定义如下:

Qt::Alignment alignment()
void setAlignment(Ot::Alignment)

alignment属性值是Qt:Alignment类型Qt帮助文档中显示的Qt::Alignment信息有如下表示

enum Qt::AlignmentFlag  //枚举类型
flags Qt::Alignment     //标志类型

第一行代码翻译一下就是Qt命名空间下的有一个变量名字叫做AlignmentFlag他是枚举类型

这表示Qt::Alignment是QFlags<Qt::AlignmentFlag>类型,但是Qt中并没有定义实际的类型Qt::Alignment,也就是不存在如下的定义:

这样的定义实际不存在

typedef QFlags<Qt::AlignmentFlag> Qt::Alignment;

Qt::AlignmentFlag 是枚举类型,其有一些枚举常量。

如:

详见Qt文档。

Ot::Alignment是一个或多个Qt:AlignmentFlag类型枚举值的组合,是一种特性标志。

即Alignment可以是 AlignLeft | AlignRight

也可以是 AlignHCenter | AlignLeft所以我们把Qt::Alignment称为枚举类型Qt::AlignmentFlag的标志类型。

给窗口上的OLabel组件label 设置对齐方式,可以使用如下的代码

ui->label->setAlignment(Qt::AlignLeft lQt::AlignVCenter);

这实际上是创建了一个Qt::Alignment类型的临时变量,相当于如下的代码:

 

QFlags<Qt::AlignmentFlag> flags = Qt::AlignLeft | Qt::AlignVCenter;
ui->label->setAlignment(flags);

看这段代码并有没有写Qt::Alignment = Qt::AlignLeft | Qt::AlignVCenter;

也可以看出实际并没有Qt::Alignment,他只是一个标志。

QFlags 类支持或、与、异或等位运算,所以也可以这样写代码\

QFlags<Qt::AlignmentFlag> flags= ui->label->alignment();//获取alignment 属性值
flags = flags | Qt::AlignVCenter;//增加垂直对齐
u1->label->setAlignment(flags);//设置alignment 属性值

这里主要是要区分帮助文档中enum Qt:AlignmentFlag 和flags Qt::Alignment的意义,不要把QLabel的setAlignment()函数的输入参数认为是枚举类型,它实际上是标志类型。

QFlags类有一个函数testFlag()可以测试某个枚举值是否包含在此标志变量中,例如:

 

//获取alignment 属性值//是否包含
QFlags<Ot;:AlignmentFlag> flags= ui->label->alignment();
bool isLeft = flags.testFlag(Qt::AlignLeft);

Qt使用QFlags来提供类型安全性。

如果要对自己的枚举类型使用QFlags,应使用Q_DECLARE_FLAGS()和Q_DECLARE_OPERATORS_FOR_FLAGS()。

例:
  class MyClass
  {
  public:
    enum Orientation
    {
        Up = 1,
        Down = 2,
        Left = 4,
        Right = 8,
    };
    Q_DECLARE_FLAGS(Orientations, Orientation)
      ...
  };
  Q_DECLARE_OPERATORS_FOR_FLAGS(MyClass::Orientations)

这样为枚举Orientation创建了一个Flags:Orientations,这个Orientations的类型就是QFlags<MyClass::Orientation>。可以用Orientations对象接收逻辑运算的值了:

Orientations f = Orientation::Up | Orientation::Down;

总结:

个人觉得QFLags其实就是一个存储枚举组合值得东西

大家如果想要在深入了解,可以去看下面这篇博客

采用博客内容:

版权声明:本文为CSDN博主「友善啊,朋友」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/kenfan1647/article/details/114801093

目录
相关文章
|
12天前
|
安全 C++
Qt moveToThread
Qt moveToThread
24 0
|
4月前
|
Linux 定位技术 C++
【Qt】-学Qt前的准备
【Qt】-学Qt前的准备
|
开发框架 Linux API
2023-6-1-Qt是什么
2023-6-1-Qt是什么
96 0
|
存储
Qt之QLCDNumber
Qt之QLCDNumber
347 0
|
Linux Android开发 C++
Qt资料大全
简述 发福利了、发福利了、发福利了,重要的事情说三遍。。。 为了方便更多Qter了解、学习Qt,现将相关资源进行整理,主要内容包括:Qt官网、编码风格、GitHub &amp; Third-Party、社区论坛、博客、书籍等。 满满的都是干货,独乐乐不如众乐乐。。。 简述 Qt官网 编码风格 GitHub Third-Party 社区论坛 博客 书籍 更多
2874 0
|
安全 并行计算
Qt之QFutureWatcher
简述 QFuture 表示异步计算的结果,QFutureWatcher 则允许使用信号和槽监视 QFuture,也就是说,QFutureWatcher 是为 QFuture 而生的。 简述 详细描述 基本使用 更多参考 详细描述 QFutureWatcher 提供了有关 QFuture 的信息和通知,使用 setFuture() 函数开始监视一个特
3673 0
|
缓存 Unix Windows
Qt之QLocalSocket
简述 QLocalSocket类提供了一个本地socket。 在Windows中,这是一个命名管道;在Unix中,这是一个本地网域socket。 如果发生错误,socketError()会返回错误的类型,errorString()则返回人类可读的错误描述。 虽然QLocalSocket是一个事件循环使用而设计,它也可以不被如此使用。这种情况下,必须使用 waitF
2638 0
|
网络协议
Qt之QHostAddress
简述 QHostAddress类提供一个IP地址。 这个类提供一种独立于平台和协议的方式来保存IPv4和IPv6地址。 QHostAddress通常与QTcpSocket、QTcpServer、QUdpSocket一起使用,来连接到主机或建立一个服务器。 可以通过setAddress()来设置一个主机地址,使用toIPv4Address()、toIPv6Address
3571 0
|
存储
Qt之findChild
简述 在Qt编程过程中,通常会有多个部件嵌套,而大多数部件都有父子依赖关系,但是有些情况下不能直接引用子部件,这时我们可以通过父部件来findChild -“查找孩子”。 简述 查找选项 findChild 描述 示例 分析 效果 源码 可能情况 查找选项 枚举Qt::FindChildOption: Qt::FindChildOp
2337 0