原文地址:http://blog.163.com/net_worm/blog/static/1277024192010097430321/

在开始分析之前交代一下,一是分析的QT在Window平台实现(其它OS类似);二、分析的手段为看源码+单步跟踪。有时候会让编译器产生预编译后的输出(使用-E参数),便于观察;三、分析得QT版本为4.5.3

下面是QT经典的Hello world程序

 1 #include <qapplication.h>
2 #include <qpushbutton.h>
3 #include <qfont.h>
4
5 int main( int argc, char **argv )
6 {
7 QApplication a( argc, argv );
8 QPushButton quit( "Quit", 0 );
9 quit.resize( 75, 30 );
10 quit.setFont( QFont( "Times", 18, QFont::Bold ) );
11 QObject::connect( &quit, SIGNAL(clicked()), &a, SLOT(quit()) );
12 quit.show();
13 return a.exec();
14 }

第一句声明了一个QApplication的对象,QApplication的类继承关系为:

1 QApplication : QCoreApplication : QObject

先观察QApplication的构造函数:

1 QApplication::QApplication(int &argc, char **argv)
2 : QCoreApplication(*new QApplicationPrivate(argc, argv, GuiClient))
3 { Q_D(QApplication); d->construct(); }

这里先后做了三件事:新生成一个QApplicationPrivate对象并传递给QCoreApplication;宏Q_D;调用d->construct()。

看Q_D的定义:

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

Q_D(QApplication);展开后是:

1 QApplicationPrivate * const d = d_func();

在附近正好看到Q_Q的定义:

1 #define Q_Q(Class) Class * const q = q_func()

由此看到在QT的程序里最好不要定义d、q这样的变量。

所以d->construct()调用的其实是

1 QApplicationPrivate::construct();

那么这个construct()干了什么事呢?

 1 void QApplicationPrivate::construct()
2 {
3 initResources(); // 初始化资源
4
5 qt_is_gui_used = (qt_appType != QApplication::Tty);
6 process_cmdline(); // 扫描分析命令行参数
7 // Must be called before initialize()
8 qt_init( this, qt_appType ); // 在initialize()之前必须执行的初始化,例如色彩、字体、键盘等
9 initialize();
10 eventDispatcher->startingUp();
11
12 }

其中eventDispatcher->startingUp();实际调用的是:QEventDispatcherWin32::startingUp(),主要是注册事件分发器。

回来看QCoreApplication的构造过程,把QApplicationPrivate对象传递给QOjbec,然后调用init()。

1 QCoreApplication::QCoreApplication(QCoreApplicationPrivate &p)
2 : QObject(p, 0)
3 {
4 init();
5 // note: it is the subclasses' job to call
6 // QCoreApplicationPrivate::eventDispatcher->startingUp();
7 }

我们先来看QCoreApplication::init()干了什么事呢?

 1 void QCoreApplication::init()
2 {
3 Q_D(QCoreApplication);
4
5 // Get the application name/instance if qWinMain() was not invoked
6 set_winapp_name(); // 设置应用程序的名字
7
8 Q_ASSERT_X(!self, "QCoreApplication", "there should be only one application object");
9 QCoreApplication::self = this;
10
11 QThread::initialize(); // 初始化线程
12
13 // use the event dispatcher created by the app programmer (if any)
14 if (!QCoreApplicationPrivate::eventDispatcher)
15 QCoreApplicationPrivate::eventDispatcher = d->threadData->eventDispatcher;
16 // otherwise we create one
17 if (!QCoreApplicationPrivate::eventDispatcher)
18 d->createEventDispatcher(); // 生成事件分发
19 Q_ASSERT(QCoreApplicationPrivate::eventDispatcher != 0);
20
21 if (!QCoreApplicationPrivate::eventDispatcher->parent())
22 QCoreApplicationPrivate::eventDispatcher->moveToThread(d->threadData->thread);
23
24 d->threadData->eventDispatcher = QCoreApplicationPrivate::eventDispatcher;
25
26 if (!coreappdata()->app_libpaths) {
27 // make sure that library paths is initialized
28 libraryPaths(); // 确认或设定DLL库的路径
29 } else {
30 d->appendApplicationPathToLibraryPaths();
31 }
32
33 qt_startup_hook(); // 目前是空函数
34 }

Object的构造又做了什么事情呢?

QObject::QObject(QObjectPrivate &dd, QObject *parent)     : d_ptr(&dd) 将QApplicationPrivate对象传递给d_ptr,这个d_ptr是什么呢?

我们看QObject的类定义,d_ptr是QObject中唯一的数据成员:

protected:     QObjectData *d_ptr; 至此,QApplication对象的初始基本分析完毕,除了很多初始化的动作之外,主要就是把QApplication和QApplicationPrivate关联起来.

转自:http://www.cnblogs.com/lfsblack/p/5278987.html

1、QT分析之QApplication的初始化的更多相关文章

  1. QT分析之QApplication的初始化

    原文地址:http://blog.163.com/net_worm/blog/static/1277024192010097430321/ 在开始分析之前交代一下,一是分析的QT在Window平台实现 ...

  2. 2、QT分析之QPushButton的初始化

    原文地址:http://blog.163.com/net_worm/blog/static/127702419201001003326522/ 在简单的QT程序的第二行,声明了一个QPushButto ...

  3. QT分析之QPushButton的初始化

    原文地址:http://blog.163.com/net_worm/blog/static/127702419201001003326522/ 在简单的QT程序的第二行,声明了一个QPushButto ...

  4. Qt事件分发机制源码分析之QApplication对象构建过程

    我们在新建一个Qt GUI项目时,main函数里会生成类似下面的代码: int main(int argc, char *argv[]) { QApplication application(argc ...

  5. 10、QT分析之WebKit

    该文章整理自 网易博客 http://blog.163.com/net_worm/blog/static/12770241920101831312381/ 转载请注明出处 WebKit是QT4新整合的 ...

  6. 3、QT分析之消息事件机制

    原文地址:http://blog.163.com/net_worm/blog/static/127702419201001432028526/ 上回我们分析到QPushButton的初始化,知道了Wi ...

  7. 5、QT分析之网络编程

    原文地址:http://blog.163.com/net_worm/blog/static/127702419201002842553382/ 首先对Windows下的网络编程总结一下: 如果是服务器 ...

  8. QT分析之WebKit

    该文章整理自 网易博客 http://blog.163.com/net_worm/blog/static/12770241920101831312381/ 转载请注明出处 WebKit是QT4新整合的 ...

  9. QT分析之网络编程

    原文地址:http://blog.163.com/net_worm/blog/static/127702419201002842553382/ 首先对Windows下的网络编程总结一下: 如果是服务器 ...

随机推荐

  1. WCF返回null超时

    Message.CreateMessage(msg.Version, msg.Headers.Action + "Response", DealObject("错误信息& ...

  2. linux:文件及文件夹管理

    http://blog.csdn.net/pipisorry/article/details/39854265 查看用户的信息 pika:~$id pikauid=1000(pika) gid=100 ...

  3. Android各版本代号、版本号、API/NDK级别、发布时间及市场份额

    Android各版本代号.版本号.API/NDK级别.发布时间及市场份额 代号 版本号 API/NDK级别 发布时间 - O 8.0 API level 26 2017-3-21 牛轧糖 Nougat ...

  4. 升级java编译器

    [Help]-[Eclipse Markerplace],搜索“Java 8”,安装插件.

  5. 正确理解web交互中的cookie与session

    cookie存储在客户端的纯文本文件 用户请求服务器脚本 脚本设置cookie内容 并 通过http-response发送cookie内容到客户端并保存在客户端本地 客户端再次发送http请求的时候会 ...

  6. Android开发中adb命令的常用方法

    Android的SDK中提供了很多有用的工具,在开发过程中如果能熟练使用这些工具,会让我们的开发事半功倍.adb是SDK提供的一个常用的命令行工具,全称为Android Debug Bridge,用于 ...

  7. Windows API 错误码

    在多数情况下,windows API在发生错误时很少抛出异常,多数是通过函数返回值进行处理.(windows api中无返回值的函数很少.) windows api错误处理通常按照以下方式:首先api ...

  8. 【转】mysql 索引过长1071-max key length is 767 byte

    问题 create table: Specified key was too long; max key length is 767 bytes 原因 数据库表采用utf8编码,其中varchar(2 ...

  9. 【机器学习】粗糙集属性约简算法与mRMR算法的本质区别

    1. 粗糙集属性约简算法仅仅选出属性重要度大的条件加入约减中,没有考虑约简中条件属性相互之间的冗余性,得到的约简往往不是都必要的,即含有冗余属性. 2. mRMR算法则除了考虑特征与类别之间的相关性, ...

  10. 【Ubuntu】任务管理器loadruner

    linux1 准备工作   可以通过两种方法验证服务器上是否配置了rstatd守护程序:    ①使用rup命令,它用于报告计算机的各种统计信息,其中就包括rstatd的配置信息.使用命令rup 10 ...