简体中文 English Български Español
Table of Content
什么是d-pointer
如果你曾经查看过Qt的源代码文件,例如 这个 [qt.gitorious.com], 你会发现里面有很多的 Q_D 和 Q_Q 宏定义. 本文就来揭开这些宏使用的目的。
Q_D 和 Q_Q 宏定义是d-pointer((也被称为 opaque pointer [en.wikipedia.org]))设计模式的一部分, 它可以把一个类库的实施细节对使用的用户隐藏, 而且对实施的更改不会打破二进制兼容。
什么是二进制兼容?
在设计像Qt这样的类库的时候,理想的行为应该是动态连接到Qt的应用程序,甚至在Qt类库升级或者替换到另外一个版本的时候,不需要重新编译就可以继续运行。例如,你的应用程序_CuteApp_是基于Qt 4.5的,你应该可以把你的Qt有4.5升级到Qt 4.6(在Windows下用安装程序,在Linux下通常有包管理器自动完成),而你用Qt4.5构建的CuteApp应该还可以运行。
什么会打破二进制兼容
那么,什么时候类库的变化需要应用程序的重新编译呢? 我们来看一个简单的例子:
- class Widget {
- ...
- private:
- Rect m_geometry;
- };
- class Label : public Widget {
- ...
- String text() const { return m_text; }
- private:
- String m_text;
- };
这里,我们有一个Widget, 包含一个geometry作为成员变量。 我们编译Widget并且发布它为*WidgetLib 1.0*。
对于 *WidgetLib 1.1*, 有人有了新的主意要添加样式表支持。没问题,我们只需要添加一个新的方法和一个新的 _数据成员_。
- class Widget {
- ...
- private:
- Rect m_geometry;
- String m_stylesheet; // NEW in WidgetLib 1.1
- };
- class Label : public Widget {
- public:
- ...
- String text() const { return m_text; }
- private:
- String m_text;
- };
我们发布WidgetLib 1.1,仅仅包含上面的变化,却发现和WidgetLib 1.0一起编译并且运行正常的 CuteApp光荣地崩溃了!
为什么会崩溃?
究其原因,通过添加了一个新的数据成员,我们最终改变了Widget 和 Label 对象的大小。 为什么会这样? 因为当你的C++编译器生成代码的时候,他会用`偏移量`来访问对象的数据。
下面是一个POD对象在内存里面布局的一个简化版本。
| Label对象在WidgetLib 1.0的布局 | Label对象在WidgetLib 1.0的布局 |
| m_geometry <偏移量 0> | m_geometry <偏移量 0> |
| ——————— | m_stylesheet <偏移量 1> |
| m_text <偏移量 1> | ————————- |
| ———————- | m_text <偏移量 2> |
在 !WidgetLib 1.0, Lable的text成员在(逻辑)偏移量为1的位置。在编译器生成的代码里,应用程序的方法Label::text()被翻译成访问Lable对象里面偏移量为1的位置。
在In WidgetLib 1.1, Label的’‘text’‘成员的(逻辑)偏移量被转移到了2的位置! 由于应用程序没有重新编译,它仍然认为text在偏移量1的位置,结果却访问了
stylesheet变量!
对刚刚入门C++的新兵 :-)
我确信,这个时候,会有人问,为什么Label::text()的偏移量的计算的代码会在CuteApp二进制文件结束,而不是在WidgetLib的二进制文件。 答案是因为Label::text() 的代码定义在头文件里面,最终被 内联 [en.wikipedia.org]。
那么,如果Label::text() 没有定义为内联函数,情况会改变吗? 这么讲,Label::text() 被移到源文件里面? 嗯,不会。C++编译器依赖对象大小在编译时和运行时相同。比如,堆栈的winding/unwinding – 如果你在堆栈上创建了一个Label对象, 编译器产生的代码会根据Label对象在编译时的大小在堆栈上分配空间。 由于Label的大小在WidgetLib 1.1运行时已经不同,Label的构造函数会覆盖已经存在的堆栈数据,最终破坏堆栈。
不要改变导出的C++类的大小
总之,一旦你的类库发布了,永远不要改变_导出的_C++类的大小或者布局(不要移动成员)。C++编译器生成的代码会假定,一个类的大小和成员的顺序_编译后_就不会改变.
那么,如何在不改变对象的大小的同时添加新的功能呢?
d-pointer
诀窍是通过保存唯一的一个指针而保持一个类库所有公共类的大小不变。这个指针指向一个包含所有数据的私有的(内部的)数据结构。内部结构的大小可以增大或者减小,而不会对应用程序带来副作用,因为指针只会被类库里面的代码访问,从应用程序的视角来看,对象的大小并没有改变 – 它永远是指针的大小。 这个指针被叫做 _d-pointer_。
这个模式的精神可以由下面的代码来概述。
- /* widget.h */
- // 前置声明. 定义在 widget.cpp或者
- // 单独的一个文件,比如widget_p.h
- class WidgetPrivate;
- class Widget {
- ...
- Rect geometry() const;
- ...
- private:
- // d-pointer never referenced in header file.
- // Since WidgetPrivate is not defined in this header,
- // any access will be a compile error
- WidgetPrivate *d_ptr;
- };
- /* widget_p.h */ (_p 意味着私有)
- struct WidgetPrivate {
- Rect geometry;
- String stylesheet;
- };
- /* widget.cpp */
- #include "widget_p.h"
- Widget::Widget()
- : d_ptr(new WidgetPrivate) // 创建私有数据{
- }
- Rect Widget::geoemtry() const {
- // d-ptr 仅仅被类库代码访问
- return d_ptr->geometry;
- }
- /* label.h */
- class Label : public Widget {
- ...
- String text();
- private:
- // 每个类维护自己的d-pointer
- LabelPrivate *d_ptr;
- };
- /* label.cpp */
- // 和WidgetPrivate不同, 我们在它自己的源文件里定义LabelPrivate
- struct LabelPrivate {
- String text;
- };
- Label::Label()
- : d_ptr(new LabelPrivate) {
- }
- String Label::text() {
- return d_ptr->text;
- }
有了上面的机构,CuteApp从来不需要直接访问d-pointer。由于_d-pointer_只是在WidgetLib被访问,而WidgetLib在每一次发布都被重新编译,私有的类可以随意的改变而不会对CuteApp带来影响。
d-pointer的其它好处
这里不全都是和二进制兼容有关。d-pointer还有其它的好处:
- 隐藏了实现细节 – 我们可以只发布带有头文件和二进制文件的WidgetLib。源文件可以是闭源代码的。
- 头文件很干净,不包含实现细节,可以直接作为API参考。The header file is clean of implementation details and can serve as the API reference.
- 由于实施需要的包含的头文件从头文件里已到了实施(源文件)里面,编译速更快。(译:降低了编译依赖)
事实上,上边的好处是微乎其微的。Qt使用d-pointer的真正原因是为了二进制兼容和Qt封闭源代码的事实.(译:Qt好像没有封闭源代码)
The q-pointer
到目前为止,我们仅仅看到的是作为C风格的数据机构的d-pointer。实际上,它可以包含私有的方法(辅助函数)。例如,LabelPrivate 可以有一个getLinkTargetFromPoint() 辅助函数,当鼠标点击的时候找到目标链接。在很多情况下,这些辅助函数需要访问公有类,也就是Lable或者它的父类Widget的一些函数。比如,一个辅助函数setTextAndUpdateWidget() 想要调用一个安排重画Widget的公有方法Widget::update()。所以,WidgetPrivate存储了一个指向公有类的指针,称为q-pointer。修改上边的代码引入q-pointer,我们得到下面代码:
- /* widget.h */
- // 前置声明. 定义在 widget.cpp
- // 或者单独的一个文件,比如widget_p.h
- class WidgetPrivate;
- class Widget {
- ...
- Rect geometry() const;
- ...
- private:
- // d-pointer 从来不在头文件被引用
- // 因为WidgetPrivate没有在头文件里定义,
- // 所有对它的访问都会带来编译错误
- WidgetPrivate *d_ptr;
- };
- /* widget_p.h */ (_p意味着私有)
- struct WidgetPrivate {
- // constructor that initializes the q-ptr
- WidgetPrivate(Widget *q) : q_ptr(q) { }
- Widget *q_ptr; // q-ptr that points to the API class
- Rect geometry;
- String stylesheet;
- };
- /* widget.cpp */
- #include "widget_p.h"
- // create private data. pass the 'this' pointer to initialize the q-ptr
- Widget::Widget()
- : d_ptr(new WidgetPrivate(this)) {
- }
- Rect Widget::geoemtry() const {
- // the d-ptr is only accessed in the library code
- return d_ptr->geometry;
- }
- /* label.h */
- class Label : public Widget {
- ...
- String text() const;
- private:
- LabelPrivate *d_ptr; // each class maitains it's own d-pointer
- };
- /* label.cpp */
- // Unlike WidgetPrivate, we define LabelPrivate in the source file
- struct LabelPrivate {
- LabelPrivate(Label *q) : q_ptr(q) { }
- Label *q_ptr;
- String text;
- };
- Label::Label()
- : d_ptr(new LabelPrivate(this)) {
- }
- String Label::text() {
- return d_ptr->text;
- }
进一步优化
对于上边的代码,创建一个Label会带来LabelPrivate和WidgetPrivate的内存分配。如果我们在Qt里面采用这种策略,对已一些像QListWidget的类,情况会相当糟糕 – 它有6层的继承层次,也就会带来最多6次的内存分配。
通过对我们的_私有_类添加一个继承层次,解决了这个问题,这样类实例化时将一个d-pointer层层向上传递。
- /* widget.h */
- class Widget {
- public:
- Widget();
- ...
- protected:
- // only sublasses may access the below
- Widget(WidgetPrivate &d); //允许子类通过他们自己的实体私有对象来初始化
- WidgetPrivate *d_ptr;
- };
- /* widget_p.h */ (_p means private)
- struct WidgetPrivate {
- WidgetPrivate(Widget *q) : q_ptr(q) { } // 构造函数初始化q-ptr
- Widget *q_ptr; // 指向API类的
- Rect geometry;
- String stylesheet;
- };
- /* widget.cpp */
- Widget::Widget()
- : d_ptr(new WidgetPrivate(this)) {
- }
- Widget::Widget(WidgetPrivate &d)
- : d_ptr(&d) {
- }
- /* label.h */
- class Label : public Widget {
- public:
- Label();
- ...
- protected:
- Label(LabelPrivate &d); // 允许Label的子类传递自己的私有数据
- //注意Label没有d_ptr!它用了父类Widgetde d_ptr.
- };
- /* label.cpp */
- #include "widget_p.h" // so we can access WidgetPrivate
- class LabelPrivate : public WidgetPrivate {
- public:
- String text;
- };
- Label::Label()
- : Widget(*new LabelPrivate) // 用我们自己的私有对象来初始化 d-pointer {
- }
- Label::Label(LabelPrivate &d)
- : Widget(d) {
- }
是不是很漂亮?现在当我们创建一个Label对象时,它会创建一个LabelPrivate(它继承了WidgetPrivate)。它把一个_d-pointer_实体传递给Widget的保护的构造函数。 Label也有这样一个保护的构造函数,可以被继承 Label的类提供自己的私有类来使用。
把q-ptr和d-ptr转型到正确的类型(QPTR和DPTR)
上一步优化的一个副作用是q-ptr和d-ptr的类型分别是Widget和WidgetPrivate。
这就意味着下面的代码不能工作。
- void Label::setText(const String &text) {
- // 不工作。虽然d_ptr指向LabelPrivate对象,但是它是WidgetPrivate类型
- d_ptr->text = text;
- }
因此,在子类里访问d-pointer的时候,需要用static_cast转型到合适的类型。
- void Label::setText(const String &text) {
- LabelPrivate *d = static_cast<LabelPrivate *>(d_ptr); // cast to our private type
- d->text = text;
- }
代码里到处都是static_cast看起来不是那么漂亮,所以我们定义了下面的宏,
- // global.h (macros)
- #define DPTR(Class) Class##Private *d = static_cast<Class##Private *>(d_ptr)
- #define QPTR(Class) Class *q = static_cast<Class *>(q_ptr)
- // label.cpp
- void Label::setText(const String &text) {
- DPTR(Label);
- d->text = text;
- }
- void LabelPrivate::someHelperFunction() {
- QPTR(label);
- q->selectAll(); // we can call functions in Label now
- }
Qt中的d-pointers
在Qt中,几乎所有的公有类都使用了d-pointer。唯一不用的情况是如果事先知道某个类永远不会添加额外的成员变量。例如,像QPoint, QRect这些类,我们不期望有新的成员添加,因此它们的数据成员直接保存在类里而没用d-pointer。
- 在Qt中,所有私有对象的基类是
QObjectPrivate Q_D和Q_Q宏提供了上边讨论的QPTR 和 DPTR的功能.- Qt的公有类有一个
Q_DECLARE_PRIVATE的宏。这个宏的代码:
- // qlabel.h
- private:
- };
- // qglobal.h
- #define Q_DECLARE_PRIVATE(Class) \
- inline Class##Private* d_func() { return reinterpret_cast<Class##Private *>(qGetPtrHelper(d_ptr)); } \
- inline const Class##Private* d_func() const { \
- return reinterpret_cast<const Class##Private *>(qGetPtrHelper(d_ptr)); } \
- friend class Class##Private;
QLabel提供了一个函数d_func()允许访问它的私有内部类。这个方法本身是私有的(因为这个宏定义在qlabel.h私有区域)。 但是 d_func() 可以被 QLabel的 朋友 (C++ 友元)调用。这主要对一些Qt类想获得QLabel的无法通过公有API访问的一些信息有用。例如,QLabel可能要跟踪用户点击了一个链接多少次。但是没有公有API访问这个信息。QStatistics是需要这个信息的一个类。Qt开发人员可以添加QStatistics作为QLabel的一个友元类,这样QStatistics 就可以label->d_func()->linkClickCount来访问。
d_func还有一个优点是保证了const正确性:在MyClass的一个const成员函数里,你需要 Q_D(const MyClass),因此你只能调用MyClassPrivate的const成员函数。如果用 “naked“的d_ptr你可以调用非const函数。
还有一个Q_DECLARE_PUBLIC宏做相反的事情。

