概述
如果程序从一个以前版本的库动态链接到新版本的库之后,能够继续正常运行,而不需要重新编译,那么我们就说这个库是二进制兼容的。
如果一个程序需要重新编译来运行一个新版本的库,但是不需要对程序的源代码进一步的修改,这个库就是源代码兼容的。
如果不确保版本之间的二进制兼容性,人们将被迫提供静态链接的二进制文件。静态二进制文件不好,因为它们浪费资源(尤其是内存),不能让程序从库中错误修正或扩展中受益。
要保证二进制兼容性,就得尽量让类不包含私有变量,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)
......
protected:
QScopedPointer<QObjectData> d_ptr;
}
class Q_CORE_EXPORT QObjectData {