详解QT源码之Qt事件机制原理

更新时间:2023-06-30 09:08:05 阅读: 评论:0

详解QT源码之Qt事件机制原理
QT 源码之 Qt 事件机制原理是本⽂要介绍的内容,在⽤Qt写Gui程序的时候,在main函数⾥⾯最后依据都是();很多书上对这句的解释是,使 Qt 程序进⼊消息循环。下⾯我们就到exec()函数内部,来看⼀下他的实现原理。
Let's go!
⾸先来到QTDIR\src\corelib\kernel\qcoreapplication.cpp
int QCoreApplication::exec()
{
if (!QCoreApplicationPrivate::checkInstance("exec"))
return -1;
//获取线程数据
QThreadData *threadData = lf->d_func()->threadData;
//判断是否在主线程创建
if (threadData != QThreadData::current()) {
qWarning("%s::exec: Must be called from the main thread", lf->metaObject()->className());
return -1;
}
//判断eventLoop是否已经创建
if (!threadData->eventLoops.isEmpty()) {
qWarning("QCoreApplication::exec: The event loop is already running");
return -1;
}
threadData->quitNow = fal;
QEventLoop eventLoop;
lf->d_func()->in_exec = true;
//创建eventLoop
int returnCode = ();
threadData->quitNow = fal;
if (lf) {
lf->d_func()->in_exec = fal;
//退出程序
emit lf->aboutToQuit();
ndPostedEvents(0, QEvent::DeferredDelete);
}
executivesreturn returnCode;
}
再来到qeventloop.cpp中。
int QEventLoop::exec(ProcessEventsFlags flags)
{
Q_D(QEventLoop);
if (d->threadData->quitNow)
return -1;
//已经调⽤过exec了。
if (d->inExec) {
qWarning("QEventLoop::exec: instance %p has already called exec()", this);
return -1;
}
d->inExec = true;
d->exit = fal;
++d->threadData->loopLevel;
//将事件类对象压⼊线程结构体中
d->threadData->eventLoops.push(this);
// remove posted quit events when entering a new event loop
// 这句不⽤翻译了把!
if (qApp->thread() == thread())
QCoreApplication::removePostedEvents(qApp, QEvent::Quit);
#if defined(QT_NO_EXCEPTIONS)
while (!d->exit)
//这⾥才是关键,我们还要继续跟踪进去。
processEvents(flags | WaitForMoreEvents);
#el
try {
while (!d->exit)
processEvents(flags | WaitForMoreEvents);
} catch (...) {
//如果使⽤了EXCEPTION,则继续对下⼀条时间进⾏处理。
qWarning("Qt has caught an exception thrown from an event handler. Throwing\n"
"exceptions from an event handler is not supported in Qt. You must\n"
"reimplement QApplication::notify() and catch all exceptions there.\n");
throw;
}
#endif
//退出eventloop前,将时间对象从线程结构中取出。
QEventLoop *eventLoop = d->threadData->eventLoops.pop();
Q_ASSERT_X(eventLoop == this, "QEventLoop::exec()", "internal error");
Q_UNUSED(eventLoop); // --relea warning
d->inExec = fal;
天津形象设计学校
--d->threadData->loopLevel;
/
/退出事件循环。
return d->returnCode;
}
日神来到了processEvents函数:
bool QEventLoop::processEvents(ProcessEventsFlags flags)
{
Q_D(QEventLoop);
//判断事件分派器是否为空。
if (!d->threadData->eventDispatcher)
return fal;
if (flags & DeferredDeletion)
QCoreApplication::ndPostedEvents(0, QEvent::DeferredDelete);
//调⽤不同平台下的事件分派器来处理事件。
return d->threadData->eventDispatcher->processEvents(flags);
guardrail
}
processEvents是在QAbstractEventDispatcher类中定义的纯虚⽅法。在QEventDispatcherWin32类有processEvents的实现。bool QEventDispatcherWin32::processEvents(QEventLoop::ProcessEventsFlags flags)
{
Q_D(QEventDispatcherWin32);
//内部数据创建。registerClass注册窗⼝类,createwindow创建窗体。
//注册socket notifiers,启动所有的normal timers
if (!d->internalHwnd)
createInternalHwnd();
d->interrupt = fal;
emit awake();
bool canWait;
bool retVal = fal;从零基础学会计
do {
QCoreApplicationPrivate::ndPostedEvents(0, 0, d->threadData);
DWORD waitRet = 0;
HANDLE pHandles[MAXIMUM_WAIT_OBJECTS - 1];
QVarLengthArray<MSG> procesdTimers;
while (!d->interrupt) {
DWORD nCount = d-&unt();
president
Q_ASSERT(nCount < MAXIMUM_WAIT_OBJECTS - 1);
MSG msg;
bool haveMessage;
if (!(flags & QEventLoop::ExcludeUrInputEvents) && !d->queuedUrInputEvents.isEmpty()) {
// process queued ur input events处理⽤户输⼊事件,放⼊队列中。
haveMessage = true;
msg = d->queuedUrInputEvents.takeFirst();
} el if(!(flags & QEventLoop::ExcludeSocketNotifiers) && !d->queuedSocketEvents.isEmpty()) {  // process queued socket events  处理socket事件,放⼊队列中。
haveMessage = true;
msg = d->queuedSocketEvents.takeFirst();
} el {
//从消息队列中取消息,同PeekMessage
haveMessage = winPeekMessage(&msg, 0, 0, 0, PM_REMOVE);
if (haveMessage && (flags & QEventLoop::ExcludeUrInputEvents)
&& ((ssage >= WM_KEYFIRST
&& ssage <= WM_KEYLAST)
|| (ssage >= WM_MOUSEFIRST
&& ssage <= WM_MOUSELAST)
|| ssage == WM_MOUSEWHEEL)) {
// queue ur input events for later processing
haveMessage = fal;
d->queuedUrInputEvents.append(msg);
}
if (haveMessage && (flags & QEventLoop::ExcludeSocketNotifiers)
&& (ssage == WM_USER && msg.hwnd == d->internalHwnd)) {
// queue socket events for later processing
haveMessage = fal;
d->queuedSocketEvents.append(msg);
}
}
if (!haveMessage) {
// no message - check for signalled objects
for (int i=0; i<(int)nCount; i++)
pHandles[i] = d->winEventNotifierList.at(i)->handle();
//注册signal--slot。
waitRet = MsgWaitForMultipleObjectsEx(nCount, pHandles, 0, QS_ALLINPUT, MWMO_ALERTABLE);
if ((haveMessage = (waitRet == WAIT_OBJECT_0 + nCount))) {
// a new message has arrived, process it
continue;
}
}
//事件队列中有事件需要处理。
if (haveMessage) {
//处理timer事件
if (ssage == WM_TIMER) {
// avoid live-lock by keeping track of the timers we've already nt
bool found = fal;
for (int i = 0; !found && i < unt(); ++i) {
const MSG procesd = stData()[i];
found = (procesd.wParam == msg.wParam && procesd.hwnd == msg.hwnd && procesd.lParam == msg.lParam);
}
if (found)
continue;
procesdTimers.append(msg);
} el if (ssage == WM_QUIT) {
if (QCoreApplication::instance())
QCoreApplication::instance()->quit();
return fal;
}
//消息分发处理。
if (!filterEvent(&msg)) {
TranslateMessage(&msg);
QT_WA({
DispatchMessage(&msg);
} , {
DispatchMessageA(&msg);
});
}
} el if (waitRet >= WAIT_OBJECT_0 && waitRet < WAIT_OBJECT_0 + nCount) {
//处理signal--slot
simim
英语词典在线翻译d->activateEventNotifier(d->winEventNotifierList.at(waitRet - WAIT_OBJECT_0));
flat是什么意思
} el {
// nothing todo so break
break;
}
retVal = true;
}
// still nothing - wait for message or signalled objects
QThreadData *ddata = d->threadData;
canWait = (!retVal
&& data->canWait
&& !d->interrupt
&& (flags & QEventLoop::WaitForMoreEvents));
if (canWait) {
DWORD nCount = d-&unt();
alulaQ_ASSERT(nCount < MAXIMUM_WAIT_OBJECTS - 1);
for (int i=0; i<(int)nCount; i++)
pHandles[i] = d->winEventNotifierList.at(i)->handle();
emit aboutToBlock();
waitRet = MsgWaitForMultipleObjectsEx(nCount, pHandles, INFINITE, QS_ALLINPUT, MWMO_ALERTABLE);
emit awake();
if (waitRet >= WAIT_OBJECT_0 && waitRet < WAIT_OBJECT_0 + nCount) {
d->activateEventNotifier(d->winEventNotifierList.at(waitRet - WAIT_OBJECT_0));
retVal = true;
}
}
} while (canWait);
return retVal;
}
⼩结:关于详解 QT 源码之 Qt 事件机制原理的内容介绍完了,基本属于代码实现的内容,最后希望本⽂对你有帮助!

本文发布于:2023-06-30 09:08:05,感谢您对本站的认可!

本文链接:https://www.wtabcd.cn/fanwen/fan/90/162427.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

标签:事件   内容   线程   处理   消息   创建   机制
相关文章
留言与评论(共有 0 条评论)
   
验证码:
Copyright ©2019-2022 Comsenz Inc.Powered by © 专利检索| 网站地图