Qt为了使其动态库最大程度上实现二进制兼容,引入了d指针的概念。那么为什么d指针能实现二进制兼容呢?为了回答这个问题,首先弄清楚什么是二进制兼容?所谓二进制兼容动态库,指的是一个在老版本库下运行的程序,在不经过编译的情况下,仍然能够在新的版本库下运行;需要经过编译才能在新版本下运行,而不需要修改该程序源代码,我们就说该动态库是源代码兼容的。要使一个dll能达到二进制兼容,对于一个结构,对于一个对象,其数据模型应该不变,若有变动,比如在类中增加数据成员或删除数据成员,其结果肯定影响对象的数据模型,从而导致原有数据程员在对象数据模型里的位移发生变化,这样的话编译后的新版本库很可能使程序发生崩溃,为了使在增加和添加项后不使对象数据模型大小发生变化,一种做法是预先分配若干个保留空间,当要添加项时,使用保留项。如下:

  1. class A {
  2. private:
  3. int a;
  4. int reserved[3];
  5. };
  6. class B {
  7. private:
  8. int a;
  9. quint32 b : 1;
  10. quint32 reserved : 31;
  11. };

这样的话,当样增加项的时候,只需要利用reserved空间,这样的话,对象模型就不会改变。但是这种做法很呆板,因为你不知道未来到底会有多少扩展项,少了不满足要求,多了浪费空间。那麽有没有一种更灵活的方法呢?如下:

  1. class Data {
  2. public:
  3. int a;
  4. };
  5. class A {
  6. private:
  7. Data *d_ptr;
  8. };

将A中的成员a放入Data 中,A中放入Data的一个指针,这样的话,无论你向Data中添加多少数据,A的对象模型始终是4个字节的大小(d_ptr指针的大小),这种做法是不是比上面的做法更灵活呢?d_ptr就是我们今天所要说的d指针,Qt为了实现二进制兼容,绝大数类中都包含有这样的指针,下面我们一起来看看Qt的d指针是怎么实现的:

如上图,这个是Qt根结点的指针的一般形式,下面来看看非根结点的一般形式,

注意这里QWidge派生自QObject,它里面没有d_ptr,但是它的成员函数可以访问d_ptr,因为 d_ptr是保护成员,且它的对象模型包含 d_ptr(这是因为派生类继承父类的所有成员)。

下面我们来看看Qt对上述两种情况是怎么实现的:

qobject.h文件:

  1. QObjectData {
  2. public:
  3. QObject *q_ptr;
  4. ...
  5. };
  6. class Q_CORE_EXPORT QObject
  7. {
  8. ...
  9. Q_DECLARE_PRIVATE(QObject)
  10. public:
  11. Q_INVOKABLE explicit QObject(QObject *parent=0);
  12. virtual ~QObject();
  13. ...
  14. protected:
  15. QObject(QObjectPrivate &dd, QObject *parent = 0);
  16. ...
  17. protected:
  18. QScopedPointer<QObjectData> d_ptr;
  19. ...
  20. };

如上,在这里我算去了其他的项,只保留了于d_ptr有关的项,首先来看看Q_DECLARE_PRIVATE(QObject)是什么:

  1. #define Q_DECLARE_PRIVATE(Class) \
  2. inline Class##Private* d_func() { return reinterpret_cast<Class##Private *>(qGetPtrHelper(d_ptr)); } \
  3. inline const Class##Private* d_func() const { return reinterpret_cast<const Class##Private *>(qGetPtrHelper(d_ptr)); } \
  4. friend class Class##Private;

根据宏定义,则Q_DECLARE_PRIVATE(QObject)翻译如下:

  1. inline QObjectPrivate *d_func()
  2. {
  3. return reinterpret_cast<QObjectPrivate *>(qGetPtrHelper(d_ptr));
  4. }
  5. inline const QObjectPrivate *d_func() const
  6. {
  7. return reinterpret_cast<const QObjectPrivate *>(qGetPtrHelper(d_ptr));
  8. }
  9. friend class QObjectPrivate;

再来看看qGetPtrHelper的定义:

  1. template <typename T> static inline T *qGetPtrHelper(T *ptr)
  2. {
  3. return ptr;
  4. }

再来看QScopePointer,它类似于智能指针,这样不用关心 d_ptr的释放,当离开QScopePointer的作用范围,QScopePointer会自动释放d_ptr指向的堆内存,那麽这个指针是什么时候生成的呢?q_ptr又是什么时候赋值的呢?让我们来看看qobject.cpp的实现:

  1. QObject::QObject(QObject *parent)
  2. : d_ptr(new QObjectPrivate)
  3. {
  4. Q_D(QObject);
  5. d_ptr->q_ptr = this;
  6. ...
  7. }
  8. QObject::QObject(QObjectPrivate &dd, QObject *parent)
  9. : d_ptr(&dd)
  10. {
  11. Q_D(QObject);
  12. d_ptr->q_ptr = this;
  13. ...
  14. }

我们看第一个构造函数,对于根结点的d_ptr指向new QObjectPrivate,而QObjectPrivate派生自QObjectData,那麽Q_D(QObject)宏表示什么意思呢?

  1. #define Q_D(Class) Class##Private * const d = d_func()

Q_D(QObject);翻译如下:

  1. QObjectPrivate * const d = d_func();

不难看出Q_D(QObject);定义了一个QObjectPrivate的常量指针,指向d_func() 的返回值,而该返回值,正是d_ptr(见头文件 d_func()的定义),因此同过Q_D宏我们就可以访问d指针了。

对于第二个构造函数稍后介绍,下面来看看非根结点的d_ptr的实现情况:

头文件:

  1. class Q_CORE_EXPORT QObjectPrivate : public QObjectData
  2. {
  3. Q_DECLARE_PUBLIC(QObject)
  4. ...
  5. };
  6. class Q_GUI_EXPORT QWidgetPrivate : public QObjectPrivate
  7. {
  8. Q_DECLARE_PUBLIC(QWidget)
  9. ...
  10. };
  11. class Q_GUI_EXPORT QWidget : public QObject
  12. {
  13. ...
  14. Q_DECLARE_PRIVATE(QWidget)
  15. ...
  16. public:
  17. ...
  18. explicit QWidget(QWidget* parent = 0, Qt::WindowFlags f = 0);
  19. ...
  20. };

我们首先来看看Q_DECLARE_PUBLIC宏:

  1. #define Q_DECLARE_PUBLIC(Class)                                    \
  2. inline Class* q_func() { return static_cast<Class *>(q_ptr); } \
  3. inline const Class* q_func() const { return static_cast<const Class *>(q_ptr); } \
  4. friend class Class;

根据宏定义,Q_DECLARE_PUBLIC(QObject)翻译如下:

  1. inline QObject *q_func()
  2. {
  3. return static_cast<QObject *>(q_ptr);
  4. }
  5. inline const QObject *q_func() const
  6. {
  7. return static_cast<const QObject *>(q_ptr);
  8. }
  9. friend class QObject;

Q_DECLARE_PUBLIC(QWidget)翻译如下:

  1. inline QWidget *q_func()
  2. {
  3. return static_cast<QWidget *>(q_ptr);
  4. }
  5. inline const QWidget *q_func() const
  6. {
  7. return static_cast<const QWidget *>(q_ptr);
  8. }
  9. friend class QWidget;

注意这里的q_ptr是在QObjectData里公有声明的,QObjectPrivate,QWidgetPrivate都派生或间接派生自QObjectData,所以可以访问q_ptr。

接下来看Q_DECLARE_PRIVATE(QWidget)的翻译:

  1. inline QWidgetPrivate *d_func()
  2. {
  3. return reinterpret_cast<QWidgetPrivate *>(qGetPtrHelper(d_ptr));
  4. }
  5. inline const QWidgetPrivate *d_func() const
  6. {
  7. return reinterpret_cast<const QWidgetPrivate *>(qGetPtrHelper(d_ptr));
  8. }
  9. friend class QWidgetPrivate;

接下来看看QWidget的构造函数的实现:

  1. QWidget::QWidget(QWidget *parent, Qt::WindowFlags f)
  2. : QObject(*new QWidgetPrivate, 0)
  3. {
  4. ...
  5. }

看到QObject(*new QwidgetPrivate, 0)这里调用了QObject的第二个构造函数,将d_ptr指向new QWidgetPrivate所指向的堆内存。

http://blog.csdn.net/rabinsong/article/details/9474859

d指针在Qt上的应用及实现(d指针能实现二进制兼容)的更多相关文章

  1. d指针在Qt上的应用及实现

    Qt为了使其动态库最大程度上实现二进制兼容,引入了d指针的概念.那么为什么d指针能实现二进制兼容呢?为了回答这个问题,首先弄清楚什么是二进制兼容?所谓二进制兼容动态库,指的是一个在老版本库下运行的程序 ...

  2. d指针在Qt上的应用及实现(有图,很清楚)

    Qt为了使其动态库最大程度上实现二进制兼容,引入了d指针的概念.那么为什么d指针能实现二进制兼容呢?为了回答这个问题,首先弄清楚什么是二进制兼容?所谓二进制兼容动态库,指的是一个在老版本库下运行的程序 ...

  3. Qt之美(一):d指针/p指针详解(解释二进制兼容,以及没有D指针就会崩溃的例子。有了D指针,所使用的对象大小永远不会改变,它就是该指针的大小。这个指针就被称作D指针)good

    Translated  by  mznewfacer   2011.11.16 首先,看了Xizhi Zhu 的这篇Qt之美(一):D指针/私有实现,对于很多批评不美的同路人,暂且不去评论,只是想支持 ...

  4. Qt之美(一):d指针/p指针详解(二进制兼容,不能改变它们的对象布局)

    Translated  by  mznewfacer   2011.11.16 首先,看了Xizhi Zhu 的这篇Qt之美(一):D指针/私有实现,对于很多批评不美的同路人,暂且不去评论,只是想支持 ...

  5. Xenko基础API笔记3- Pointers指针设备屏幕上点对应的手指触摸。

    样本这里是一个简单的示例程序,跟踪目前在屏幕上的指针和打印他们的位置.访问输入字段,类继承自@ SiliconStudio.Xenko.脚本的类. public override async Task ...

  6. Qt上FFTW組件的编译与安裝

    Qt上FFTW組件的編譯安裝 FFTW是一個做頻譜非常實用的組件,本文講述在Windows和Linux兩個平臺使用FFTW組件.Windows下的的FFTW組件已經編譯好成爲dll文件,按照開發應用的 ...

  7. C++学习笔记----3.2 C++引用在本质上是什么,它和指针到底有什么区别

    从概念上讲.指针从本质上讲就是存放变量地址的一个变量,在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变. 而引用是一个别名,它在逻辑上不是独立的,它的存在具有 ...

  8. Qt之二进制兼容

    一.回顾 使用qt2年多了,但是还是觉得很陌生,总是会被qt搞的很紧张,有时候当我自信满满的打开帮助文档,搜索某个已知的类时,由于笔误敲错了一个字母而出现了另外一个类,不过奇怪的是还真有这么一个类,哎 ...

  9. Qt 中的二进制兼容策略(简而言之就是地址不能变,剩下的就是让地址不变的技巧)

    本文翻译自 Policies/Binary Compatibility Issues With C++ 二进制兼容的定义 如果程序从一个以前版本的库动态链接到新版本的库之后,能够继续正常运行,而不需要 ...

随机推荐

  1. 游戏基础元素——Cocos2d-x学习历程(八)

    1.Director:导演 从字面上理解,这是一个"导演"类,Director是控制游戏流程的主要组件.CCDirector的工作确实跟导演非常类似,主要负责以下工作: 游戏呈现方 ...

  2. 指针直接赋值为整型AND利用宏定义求结构体成员偏移量

    首先我们要更正一个很熟悉的概念,那就是指针不仅仅是“地址”,指针还有一个很重要的特性,那就是“类型”. 指针初始化时,“=”的右操作数; 除外,该语句表示指针为空): 所以 ; 这样的代码是不允许的. ...

  3. OC语法6——内存管理之引用计数器(retain,release)

    OC内存管理: 一.引用计数器: Java有垃圾回收机制(Garbage Collection,GC).也就是说当我们创建对象后,不需要考虑回收内存的事,Java的垃圾回收机制会自动销毁该对象,回收它 ...

  4. Spark 算子

    0.parallelize 1.map 2.mapValues 3.flatMap 4.mapPartitions 5.mapPartitionsWithIndex 6.filter 7.reduce ...

  5. Docker基本概念填坑

    Docker的基本概念填坑 Docker的基本概念填坑 1. Docker的基本组成 Docker Client客户端 Docker Daemon守护进程 Docker Image镜像 Docker ...

  6. codeforces 645E . Intellectual Inquiry

    题目链接 如果不考虑重复的元素, 那么我们可以很容易的发现, 长度为n的字符串它的子串数量是 $ 2^n $ . 我们设每个到位置i, 答案的数量为f[i]. 然后我们考虑重复的, 我们发现, 每加入 ...

  7. Oracle EBS-SQL (SYS-1): sysadmin_用户职责查询.sql

    select fu.user_name 用户名, fu.description 用户说明, frv.RESPONSIBILITY_NAME 职责名称, REQUEST_GROUP_NAME 报表组, ...

  8. poj2140---herd sums

    #include<stdio.h> #include<stdlib.h> int main() { ,i,j; scanf("%d",&n); ;i ...

  9. ebs清除并法管理器所清除的表

    In this Document   Goal   Solution   References Applies to: Oracle Concurrent Processing - Version 1 ...

  10. freemaker

    FreeMarker模板文件主要由如下4个部分组成:  1,文本:直接输出的部分  2,注释:<#-- ... -->格式部分,不会输出  3,插值:即${...}或#{...}格式的部分 ...