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

目录
相关文章
|
7月前
|
Linux 定位技术 C++
【Qt】-学Qt前的准备
【Qt】-学Qt前的准备
|
存储 Cloud Native 前端开发
Qt QScrollArea使用
Qt QScrollArea使用
26 QT - 事件过滤器
26 QT - 事件过滤器
48 0
|
网络协议 Java PHP
|
安全 并行计算
Qt之QFutureWatcher
简述 QFuture 表示异步计算的结果,QFutureWatcher 则允许使用信号和槽监视 QFuture,也就是说,QFutureWatcher 是为 QFuture 而生的。 简述 详细描述 基本使用 更多参考 详细描述 QFutureWatcher 提供了有关 QFuture 的信息和通知,使用 setFuture() 函数开始监视一个特
3713 0
|
数据安全/隐私保护
Qt之QUrl
简述 QUrl 类提供了一个方便的接口使用 URLs。 它可以解析和构造编码和未编码形式的 URLs。QUrl 也支持国际化域名(IDNs)。 简述 详细描述 错误检查 字符转换 URL格式 scheme Authority user info path query fragment 深入使用 相对路径 用户输入 文件名 主机端口 本地文件 百分比编码
6346 0
|
存储
Qt之QUrlQuery
简述 QUrlQuery 类提供了一种方法来操纵 URL 查询中的 key-value 对。 简述 详细描述 编码 处理空格和加号 全解码 非标准分隔符 使用 QUrlQuery 分隔符 查询 删除 是否为空 详细描述 QUrlQuery 用来解析 URL 中的查询字符串,像下面这样: 上述的查询字符串在 URL 中 被用来传输
2858 0
Qt之QScrollArea
简述 QScrollArea提供了一个滚动视图到另一个部件。 滚动区域用于显示一个画面中的子部件的内容。如果部件超过画面的大小,视图可以提供滚动条,这样就都可以看到部件的整个区域。 简述 基本使用 对齐方式 调整部件大小 手动调整 自动调整 获取与移除部件 获取 移除 基本使用 子部件必须使用setWidget()指定,例如: QLab
3244 0