原文地址: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
