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. 大数据安装之Kafka(用于实时处理的消息队列)

    一.安装部署kafka 1.集群规划 hadoop102                                 hadoop103                          hado ...

  2. Hibernate一对多

    例如:一个用户可以对应多个订单  但一个订单只能对应一个用户 1.创建实体类 2.编写映射文件 <?xml version="1.0"?> <!DOCTYPE h ...

  3. zookeeper基础学习-简介

    1.zookeeper的使命 zookeeper可以在分布式系统的协作多个任务(一个任务是指一个包含多个进程的任务),这个任务可以是为了协作或者是为了管理竞争. 协作:多个进程需要一同处理某些事情,一 ...

  4. 查看oracle是否锁表以及解决方法

    Oracle数据库操作中,我们有时会用到锁表查询以及解锁和kill进程等操作,那么这些操作是怎么实现的呢?本文我们主要就介绍一下这部分内容.(1)锁表查询的代码有以下的形式: select count ...

  5. hdu1258 dfs 给一个指定的target数和一个数列,要求不重复选择其中的数使得和为target并打印,结果不可重复。

    #include<bits/stdc++.h> using namespace std; typedef unsigned int ui; typedef long long ll; ty ...

  6. 题解 P5663 【加工零件【民间数据】】

    博客园体验更佳 讲讲我的做法 确定做法 首先,看到这道题,我直接想到的是递归,于是复杂度就上天了,考虑最短路. 如何用最短路 首先,看一张图 我们该如何解决问题? 问题:\(3\)做\(5\)阶段的零 ...

  7. 【2019沈阳网络赛】G、Special necklace——自闭的物理题

    这道题让我差点怀疑自己高考没考过物理 题意中 he measures the resistance of any two endpoints of it, the resistance values ...

  8. 进制-Iterative-进制转换

    2019-12-02 21:15:31 进制转换是计算机科学里的一个基础算法,通常可以使用如下的模版来进行计算. 下面我们来讨论一些关于进制的题目. 1271. Hexspeak  问题描述: 问题求 ...

  9. mysql-8.0.19-winx64下载

    mysql-8.0.19-winx64 下载链接 提取码:m7qp

  10. 使用PostgreSQL注意事项

    一.大小写特别敏感 大写字段需要用“”引号(pg字段名使用“”,MySQL字段名使用``) ******表名以及字段名如果是小写但是为关键字,比如name,则也需使用"": 二.分 ...