概述
如果程序从一个以前版本的库动态链接到新版本的库之后,能够继续正常运行,而不需要重新编译,那么我们就说这个库是二进制兼容的。
如果一个程序需要重新编译来运行一个新版本的库,但是不需要对程序的源代码进一步的修改,这个库就是源代码兼容的。
如果不确保版本之间的二进制兼容性,人们将被迫提供静态链接的二进制文件。静态二进制文件不好,因为它们浪费资源(尤其是内存),不能让程序从库中错误修正或扩展中受益。
要保证二进制兼容性,就得尽量让类不包含私有变量,Qt的风格就保证了这一点,不信可以看QObject的源码,你会惊讶地发现它竟然只有一个成员变量QScopedPointer<QObjectData> d_ptr;。QObject的大小是8,除了虚函数表指针需要的4个字节以外,另外的4个字节就是这个d_ptr。QObjectData是包含所有数据的结构体,这个结构体的大小可以随意改变而不会产生副作用,应用程序只使用相关的公有类即QObject,d_ptr就叫做d指针。
这种设计模式叫做句柄实体模式,也就是以QObject为基类的类一般都是句柄类,一般只有一个指针指向一个实体类,在实体类中保存全部的数据。QObject比平常的用法更复杂,涉及到QObjectData和QObjectPrivate两个类,一般只要有ClassPrivate就够了。
源码分析
相关几个类源码大致如下:
class Q_CORE_EXPORT QObject
{
Q_DECLARE_PRIVATE(QObject)
......

本文介绍了Qt库中的d指针和q指针设计模式,以保证二进制兼容性并隐藏实现细节。Q_DECLARE_PRIVATE和Q_DECLARE_PUBLIC宏定义用于声明和访问私有实体类,Q_Q和Q_D提供对d指针和q指针的简写。通过源码分析,阐述了d指针带来的优点,并给出了GitHub上的示例程序以供参考。

3496

被折叠的 条评论
为什么被折叠?



