Qt 之 Threads 和 QObjects

本文详细介绍了Qt中的线程和QObjects的使用,包括QObject的可重入性、每个线程的事件循环、从其他线程访问QObject子类以及跨线程的信号和槽。强调了QThread的事件循环、线程安全问题以及信号槽的四种连接类型,特别指出Direct Connection在不同线程间的不安全性,并提供了Mandelbrot和Blocking Fortune Client两个示例来说明实践中的应用。

作者: 一去、二三里
个人微信号: iwaleon
微信公众号: 高效程序员

QThread继承自QObject,它发射信号(signals)以表明线程执行开始或结束,并提供了一些槽函数(slots)。

更有趣的是,QObjects可以在多线程中使用,发射信号以在其它线程中调用槽函数,并且向“存活”于其它线程中的对象发送事件(post events)。这是可能的,因为每一个线程都拥有它自身的事件循环(event loop)。

QObject可重入性

QObject是可重入的。它的大多数非GUI子类,例如:QTimer、QTcpSocket、QUdpSocket和QProcess,也都是可重入的,这使得在多线程中同时使用这些类成为可能。**注意:**这些类被设计成在单一线程中创建和使用的,在一个线程中创建一个对象而在另一个线程中调用该对象的函数,不保证能行得通。需要注意有三个约束:

  • 一个QObject类型的孩子必须总是被创建在它的父亲所被创建的线程中。这意味着,除了别的以外,永远不要把QThread对象(this)作为该线程中创建的一个对象的父亲(因为QThread对象自身被创建在另外一个线程中)。

  • 事件驱动的对象可能只能被用在一个单线程中。特别是,这适用于计时器机制(timer mechanism)和网络模块。例如:你不能在不属于这个对象的线程中启动一个定时器或连接一个socket,必须保证在删除QThrea

<think>我们正在讨论Qt框架中常量的声明使用。在C++中,常量通常使用`const`关键字或预处理器宏`#define`来定义。然而,在Qt中,我们也可以使用Qt特有的机制,比如使用命名空间枚举。此外,由于Qt是C++的扩展,所以C++中定义常量的方法在Qt中同样适用。 下面我们将介绍几种在Qt中声明使用常量的方法: ### 1. 使用`const`关键字 这是C++中最常用的定义常量的方法。在Qt中,我们同样可以在类中、函数中或全局作用域中定义`const`常量。 **示例:** ```cpp // 全局常量 const int MAX_WIDTH = 1024; const int MAX_HEIGHT = 768; class MyWidget : public QWidget { public: // 类内常量 static const int DEFAULT_WIDTH = 800; static const int DEFAULT_HEIGHT = 600; void myFunction() { // 局部常量 const int localConst = 100; // ... } }; ``` 注意:对于类内的静态常量(static const),如果类型是整数(如int, char, bool等)或枚举类型,可以在类内直接初始化(如上面的例子)。对于非整数类型(如double)或者需要取地址的情况,则需要在类外进行定义(但不需再次初始化,除非是C++17以后的版本,内联变量允许在类内定义)。 ### 2. 使用枚举 在类中,我们可以使用枚举类型来定义一组相关的常量。 **示例:** ```cpp class MyClass { public: enum { MaxCount = 100, BufferSize = 1024 }; // 或者使用强类型枚举(C++11) enum class Limits : int { MaxCount = 100, BufferSize = 1024 }; }; ``` ### 3. 使用`#define`宏 虽然`#define`是预处理器宏,不是类型安全的,但在某些情况下(如条件编译)仍然被使用。在Qt中,我们也可以使用它来定义常量。 **示例:** ```cpp #define MAX_WIDTH 1024 #define MAX_HEIGHT 768 ``` 但是,由于宏没有作用域,且容易引起命名冲突,所以通常推荐使用`const`常量。 ### 4. 使用`constexpr`(C++11起) C++11引入了`constexpr`关键字,用于定义编译期常量。在Qt中,如果使用C++11或更高版本,我们可以使用`constexpr`。 **示例:** ```cpp constexpr int MAX_WIDTH = 1024; constexpr int MAX_HEIGHT = 768; class Circle { public: constexpr static double PI = 3.1415926535; }; ``` ### 5. 使用Qt的命名空间 Qt本身在全局命名空间中定义了一些常量,例如在`Qt`命名空间中有一些枚举值(如`Qt::AlignLeft`)。我们也可以在自己的命名空间中定义常量。 **示例:** ```cpp namespace MyConstants { const int MAX_WIDTH = 1024; const int MAX_HEIGHT = 768; } ``` ### 6. 使用Q_GLOBAL_STATIC宏(用于全局静态对象) 如果常量是一个对象(比如QString),并且我们希望它是全局唯一的,可以使用`Q_GLOBAL_STATIC`来创建线程安全的全局静态对象。 **示例:** ```cpp Q_GLOBAL_STATIC(QString, globalString) // 使用 QString str = *globalString(); ``` 但是,对于常量来说,通常我们使用`const`修饰的全局变量即可,因为它是只读的,所以线程安全。 ### 在Qt中使用常量 在Qt中,我们可以在任何需要的地方使用这些常量,比如在设置窗口大小时: ```cpp MyWidget::MyWidget(QWidget *parent) : QWidget(parent) { resize(MyWidget::DEFAULT_WIDTH, MyWidget::DEFAULT_HEIGHT); } ``` ### 注意事项 - 避免在头文件中定义非静态的全局常量(除非是内部链接的,如加上`static`或放在匿名命名空间内),否则当多个源文件包含该头文件时会导致重复定义。 - 推荐在头文件中使用`constexpr`、`static const`成员、枚举或者内联函数来定义常量。 - 对于字符串常量,可以使用`const char*`、`const QString`或者`QLatin1String`(用于字符串字面量)等。 ### 总结 在Qt中声明使用常量与标准C++基本相同,主要使用`const`、`constexpr`、枚举`static`成员等方式。根据常量的作用域类型选择合适的方式,并注意避免重复定义命名冲突。 §§相关问题§§ 1. Qt中如何定义全局常量? 2. 在Qt的类中,如何声明一个常量成员? 3. Qt中`constexpr``const`有什么区别? 4. 如何在Qt中使用枚举类型定义常量? 5. Qt中定义字符串常量有哪些方法?</think>### Qt框架中声明使用常量的方法 在Qt框架中声明使用常量主要遵循标准C++语法规则,同时结合Qt特有的机制。以下是几种常见方法: #### 1. **使用`const`关键字**(推荐) ```cpp // 声明常量 const int MAX_WIDTH = 1024; const double PI = 3.1415926; const QString APP_NAME = "MyQtApp"; // 类内常量 class MyClass : public QObject { Q_OBJECT public: static const int MAX_COUNT = 100; // 整型常量可直接初始化 static const QString DEFAULT_NAME; }; // 类外定义(非整型需在.cpp中定义) const QString MyClass::DEFAULT_NAME = "Default"; ``` #### 2. **使用`constexpr`**(C++11及以上) ```cpp constexpr int BUFFER_SIZE = 512; // 编译期常量 constexpr QSize DEFAULT_WINDOW_SIZE(800, 600); ``` #### 3. **使用枚举** ```cpp enum Limits { MAX_USERS = 10, TIMEOUT_MS = 5000 }; // C++11强类型枚举 enum class Config : int { MAX_THREADS = 8, CACHE_SIZE = 1024 }; ``` #### 4. **命名空间常量** ```cpp namespace AppConstants { const int VERSION_MAJOR = 1; const int VERSION_MINOR = 0; const QString LOG_PATH = "/var/log"; } // 使用 int ver = AppConstants::VERSION_MAJOR; ``` #### 5. **Qt特有的常量机制** ```cpp // Q_GLOBAL_STATIC创建线程安全单例常量 Q_GLOBAL_STATIC(QString, globalConfigPath) void initApp() { *globalConfigPath() = QStandardPaths::writableLocation(QStandardPaths::ConfigLocation); } ``` #### 使用场景示例 ```cpp // 在Qt组件中使用 void MainWindow::initUI() { resize(AppConstants::DEFAULT_WIDTH, AppConstants::DEFAULT_HEIGHT); statusBar()->showMessage(tr("Version: %1").arg(AppConstants::VERSION)); } // 信号槽连接中使用 connect(button, &QPushButton::clicked, [MAX = Limits::MAX_USERS]() { qDebug() << "Max users allowed:" << MAX; }); ``` #### 最佳实践建议: 1. **优先选择`const`或`constexpr`**:类型安全且支持作用域控制 2. **避免`#define`宏常量**:缺乏类型检查且易污染全局命名空间 3. **类内常量使用`static const`**:确保每个类只有一份实例 4. **复杂常量使用单例模式**:通过`Q_GLOBAL_STATIC`管理初始化顺序 5. **字符串常量使用`QLatin1String`**:减少内存分配(适用于静态字符串) ```cpp constexpr auto DB_NAME = QLatin1String("app_database"); ``` #### 注意事项: - 在头文件中声明常量时,若非整型需在实现文件中定义 - 线程间共享的常量需确保线程安全性(使用原子操作或互斥锁) - Qt元对象系统(如Q_PROPERTY)要求常量在编译时已知 ---
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一去丶二三里

有收获,再打赏!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值