d指针和q指针

我们在类成员名称使用d指针中,已经介绍过了d指针。

这是一个绝妙的技巧,能够在不破坏二进制兼容性的情况下将新的私有数据成员添加到类中。此外,它还能保持头文件的干净,并隐藏具体的实现,加速编译。

简单示例

// foo.h
class FooPrivate; class Foo
{
public:
Foo();
int getData() const;
void setData(int d);
private:
FooPrivate* d;
}; // foo.cpp
class FooPrivate {
public:
FooPrivate() : data(0) {}
int data;
}; Foo::Foo() : d(new FooPrivate) {} int Foo::getData() const { return d->data; } void Foo::setData(int d) { d->data = d; }

Foo类中只暴露了接口,具体的实现和数据都隐藏到了cpp文件的FooPrivate类中。

q指针

d指针,用于在公有类中访问对应的私有类。对应的,q指针,用于在私有类中反向访问器对应的公有类。

我们可以对上面的代码进行简单的修改

// foo.cpp
class FooPrivate {
public:
FooPrivate(Foo *f) : data(0), q(f) {}
int data;
Foo *q;
}; Foo::Foo() : d(new FooPrivate(this)) {}

QObject和QObjectPrivate

// qobject.h
class Q_CORE_EXPORT QObjectData {
public:
virtual ~QObjectData() = 0;
QObject *q_ptr;
QObject *parent;
QObjectList children; uint isWidget : 1;
uint blockSig : 1;
uint wasDeleted : 1;
uint isDeletingChildren : 1;
uint sendChildEvents : 1;
uint receiveChildEvents : 1;
uint isWindow : 1; //for QWindow
uint unused : 25;
int postedEvents;
QDynamicMetaObjectData *metaObject;
QMetaObject *dynamicMetaObject() const;
}; class Q_CORE_EXPORT QObject
{
Q_OBJECT
Q_PROPERTY(QString objectName READ objectName WRITE setObjectName NOTIFY objectNameChanged)
Q_DECLARE_PRIVATE(QObject) public:
Q_INVOKABLE explicit QObject(QObject *parent=Q_NULLPTR);
....
protected:
QObject(QObjectPrivate &dd, QObject *parent = Q_NULLPTR);
protected:
QScopedPointer<QObjectData> d_ptr;
....
} // qobject_p.h
class Q_CORE_EXPORT QObjectPrivate : public QObjectData
{
Q_DECLARE_PUBLIC(QObject)
...
}

QObject中的的d_ptr就是d指针,QObjectData中的q_ptr就是q指针。

qobject.h是提供给用户的可见的头文件,qobject_p.h是内部私有的头文件。

在简单示例中,我们把私有类定义在了cpp文件中,如果私有类太大,当然可以单独定义一个头文件,然后在cpp文件中进行引用。

// qobject.cpp
#include "qobject.h"
#include "qobject_p.h"
... QObject::QObject(QObject *parent)
: d_ptr(new QObjectPrivate)
{
Q_D(QObject);
d_ptr->q_ptr = this;
...
} QObject::QObject(QObjectPrivate &dd, QObject *parent)
: d_ptr(&dd)
{
Q_D(QObject);
d_ptr->q_ptr = this;
...
}

上面的构造函数的内容就不言而喻了,对d指针和q指针进行赋值。

对于第二个受保护的构造函数,具体干什么用的,留给大家自行学习了,提示一下,可以前往qwidget.cpp中进行查看,用于在派生类中对上一个基类进行赋值。

这里大家不知道发现没有,有几个特殊的宏,Q_DECLARE_PUBLIC,Q_DECLARE_PRIVATE,Q_D,其实还有Q_Q。这些宏大家可以前往qglobal.h中查看。最终的作用是为了方便访问d指针和q指针,在函数中声明Q_D或Q_Q以后,可以直接使用d和q变量来代替d指针和q指针了。

关于d指针和q指针的更多的信息,请参考https://wiki.qt.io/D-Pointer

qtcreator中的变体1

qtcreator由于使用了插件机制,相当一部分暴露出来的组件都是单例模式,使用如下模式获取句柄。

static T *instance();

因此产生了一些变体。示例如下:

// foo.h
class FooPrivate; class Foo
{
friend class FooPrivate;
public:
static Foo *instance();
private:
Foo();
~Foo();
...
}; // foo_p.h
class Foo; class FooPrivate {
public:
FooPrivate(Foo *qf) : q(qf) {}
private:
Foo *q;
...
}; // foo.cpp
#include "foo.h"
#include "foo_p.h"
... static FooPrivate *d = 0;
static Foo *m_instance = 0; Foo *Foo::instance()
{
return m_instance;
} Foo::Foo()
{
m_instance = this;
d = new FooPrivate(this);
} Foo::~Foo()
{
delete d;
d = 0;
m_instance = 0;
}

这里主要的变化在于,d指针不再是Foo的成员了,Foo和FooPrivate都是定义在cpp的静态变量,在Foo的构造函数中初始化。这样,在Foo的成员函数中,也能直接使用d指针。

qtcreator中的变体2

此外,还有一种变体。如果只想暴露接口类,供用户调用,那么可以隐藏具体的实现类,并通过其他的管理类的相关成员函数来返回接口类指针。

// foo.h
class Foo
{
public:
void algo() = 0;
} // foo_p.h
class FooPrivate : public Foo
{
public:
void algo() override;
protect:
void doAlgo() { }
} // foo.cpp
void FooPrivate::algo() { doAlgo(); }
void FooPrivate::doAlgo() { } // foomanager.h
class FooManager
{
static Foo *foo();
}

对用户只提供foo.h和foomanager.h即可,把细节和具体实现都封装起来。用户只能通过FooManager的函数获取foo句柄,并通过foo句柄调用接口。

小结

其实,只要掌握了原理,各种变化就随意了。

  1. 对于管理类来说,一般是单例模式的,这种情况下,我们可以在cpp文件中定义静态的m_instance和d,如变体1。
  2. 对于非管理类,譬如QObject,可以创建类的多个实例的,我们一般需要在公有类中把私有类指针d作为成员变量,如简单示例。

最后,提醒一点,如果在cpp中使用Q_OBJECT,请注意先使用moc工具创建xx_moc.cpp,并#include到该cpp中,否则会报错的,"undefined reference to vtable"。


原创造福大家,共享改变世界

献出一片爱心,温暖作者心灵


qt creator源码全方面分析(4-1)的更多相关文章

  1. qt creator源码全方面分析(3-3)

    目录 qtcreatordata.pri 定义stripStaticBase替换函数 设置自定义编译和安装 QMAKE_EXTRA_COMPILERS Adding Compilers 示例1 示例2 ...

  2. qt creator源码全方面分析(3-5)

    目录 qtcreatorlibrary.pri 使用实例 上半部 下半部 结果 qtcreatorlibrary.pri 上一章节,我们介绍了src.pro,这里乘此机会,把src目录下的所有项目文件 ...

  3. qt creator源码全方面分析(0)

    本人主攻C++和Qt. 上两天刚研究完Qt install framework(IFW)应用程序安装框架. google没发现有正儿八经的官方文档的翻译,我就进行了翻译哈!! 系列文章具体见:http ...

  4. qt creator源码全方面分析(4-0)

    Qt系统 Qt Creator源码是在Qt对象和框架基础下写的,因此,阅读Qt Creator源码,你首先对Qt得有一定的了解. Qt Core Qt Core特征: The Meta-Object ...

  5. qt creator源码全方面分析(4-2)

    目录 global头文件 global.h xx.h global头文件 插件的本质就是动态链接库,对于库,需要导出符号,供用户导入使用.在qt creator的源码中,存在固定的导入导出模式. gl ...

  6. qt creator源码全方面分析(4-5)

    目录 Qt中的字符串 QLatinString 详细介绍 源码 小结 QStringLiteral(str) 详细介绍 源码 小结 Qt中的字符串 Qt中处理字符串最常用的肯定是QString,但是在 ...

  7. qt creator源码全方面分析(4-6)

    目录 Qt插件基础 Qt插件基础 我们知道Qt Creator源码是基于插件架构的,那么我们先来介绍下插件基础知识. 相关内容如下: How to Create Qt Plugins [ - Defi ...

  8. qt creator源码全方面分析(3-2)

    目录 qtcreator.pri 判断重复包含 定义版本信息 VERSION 定义IDE名称 启用C++14 CONFIG 自定义函数 Replace Functions Test Functions ...

  9. qt creator源码全方面分析(2-7)

    目录 Completing Code 补全代码片段 编辑代码片段 添加和编辑片段 删除片段 重置片段 补全Nim代码 Completing Code 在编写代码时,Qt Creator建议使用属性,I ...

  10. qt creator源码全方面分析(2-10-1)

    目录 Getting and Building Qt Creator 获取Qt 获取和构建Qt Creator Getting and Building Qt Creator 待办事项:应该对此进行扩 ...

随机推荐

  1. Java锁的深度化--重入锁、读写锁、乐观锁、悲观锁

    Java锁 锁一般来说用作资源控制,限制资源访问,防止在并发环境下造成数据错误 锁作为并发共享数据,保证一致性的工具,在JAVA平台有多种实现(如 synchronized(重量级) 和 Reentr ...

  2. Swift4.1 新特性compactMap函数

    关于compactMap函数 苹果在Swift 4.1中新增compactMap函数,用来代替flatMap函数. 在Swift标准库中compactMap定义如下 public func compa ...

  3. angular的性能分析 -随记

    $watch 的实现原理和性能分析 只有双向绑定的 scope 才会被加入$watch队列,或者手动绑定$watch的$scope 所有放在 $scope 中的变量或函数都被加入到了$watch队列当 ...

  4. [Docker6] Docker compose多容器运行与管理

    六.Docker compose docker compose就是通过yml文件来定义和运行多个容器docker应用程序的工具,三步过程就能跑起一个compose: 定义应用程序的环境(yml中) 定 ...

  5. Bootstrap 基本配置与应用

    配置使用 下载文件引用 下载方式:Bootstrap官网 https://www.bootcss.com/ 引用 例: <head> <meta charset="utf- ...

  6. 搭建Hadoop集群需要注意的问题:

    搭建Hadoop集群需要注意的问题: 1.检查三台主机名是否正确 2.检查三台IP是否正确 3.检查 /etc/hosts 映射是否正确 4.检查 JDK和Hadoop 是否安装成功(看环境变量配置) ...

  7. 多GPU使用详解

    目录: 介绍 记录设备状态 手动分配状态 允许GPU内存增长 在多GPU系统是使用单个GPU 使用多个 GPU 一.介绍 在一个典型的系统中,有多个计算设备.在 TensorFlow 中支持的设备类型 ...

  8. ICCV 2019|70 篇论文抢先读,含目标检测/自动驾驶/GCN/等(提供PDF下载)

    虽然ICCV2019已经公布了接收ID名单,但是具体的论文都还没放出来,为了让大家更快得看论文,我们汇总了目前已经公布的大部分ICCV2019 论文,并组织了ICCV2019论文汇总开源项目(http ...

  9. 使用scikit-learn解决文本多分类问题(附python演练)

    来源 | TowardsDataScience 译者 | Revolver 在我们的商业世界中,存在着许多需要对文本进行分类的情况.例如,新闻报道通常按主题进行组织; 内容或产品通常需要按类别打上标签 ...

  10. 两行代码统计模型参数量与FLOPs,这个PyTorch小工具值得一试

    你的模型到底有多少参数,每秒的浮点运算到底有多少,这些你都知道吗?近日,GitHub 开源了一个小工具,它可以统计 PyTorch 模型的参数量与每秒浮点运算数(FLOPs).有了这两种信息,模型大小 ...