Qt 之 消息机制和事件讲解
1、圖形界面應用程序的消息處理模型
特點:
基于操作系統才能運行;
GUI應用程序提供的功能必須由用戶觸發;
用戶操作界面時操作系統是第一個感知的 ;
系統內核的消息通過事件處理轉變成QT的信號。
2. Qt中的事件處理
(1)在Qt中,事件被封裝成一個個對象,所有的事件均繼承自抽象類QEvent. 事件處理的核心包括事件產生、分發、接受和處理。
①事件的產生
問題:誰來產生事件?
最容易想到的是我們的輸入設備,比如鍵盤、鼠標產生的keyPressEvent,keyReleaseEvent,mousePressEvent,mouseReleaseEvent事件(他們被封裝成QMouseEvent和QKeyEvent)。
②Qt中事件的分發
問題:誰來負責分發事件?
對于non-GUI的Qt程序,是由QCoreApplication負責將QEvent分發給QObject的子類-Receiver.;對于Qt GUI程序,由QApplication來負責。
③事件的接受和處理
問題:誰來接受和處理事件?
答案是QObject。類是整個Qt對象模型的心臟,事件處理機制是QObject三大職責(內存管理、內省(intropection)與事件處理制)之一。任何一個想要接受并處理事件的對象均須繼承自QObject,可以選擇重載QObject::event()函數或事件的處理權轉給父類。
(2)Qt平臺將系統產生的消息轉變成Qt事件
A. Qt事件是一個QEvent(或子類)的對象;
B. 有時一個事件包含多個事件類型,比如鼠標事件又可以分為鼠標按下、雙擊、和移動多種操作事件類型由QEvent類的枚舉型QEvent::Type來表示,可由幫助文檔進行查詢;
C. Qt事件用于描述程序內部或外部發生的對應動作(描述的是操作系統發生來的消息,一個系統消息對應著一個消息事件);
D. 任意QObject對象都具備時間處理的能力.
note:QEvent子類可以表示一個事件,但并不能處理這個事件
Qt 程序需要在main()函數創建一個QApplication對象,然后調用它的exec()函數。這個函數就是開始?Qt 的事件循環。在執行exec()函數之后,程序將進入事件循環來監聽應用程序的事件。當事件發生時,Qt 將創建一個事件對象。Qt 中所有事件類都繼承于QEvent。在事件對象創建完畢后,Qt 將這個事件對象傳遞給QObject的event()函數。event()函數并不直接處理事件,而是將這些事件對象按照它們不同的類型,分發給不同的事件處理器(event handler)。如上所述,event()函數主要用于事件的分發。
(3)GUI應用程序的事件處理方式
A. Qt事件產生后會立即被分發到QWidget對象(QObject的子類,如按鍵QPushButton對象等);
B. QWidget對象其內部會有一個event(QEVent*)函數被調用,進行事件處理;
C. event()根據事件類型調用不同的事件處理函數(默認的子函數);
D. 在事件處理函數中發送Qt中預定義的信號;
E. 調用信號關聯的槽函數。
(4) QPushButton事件處理分析
①接收到鼠標事件
②QApplication調用QObject::event(QEvent*)成員函數來處理,進行事件的分派。
③調用QPushButton的mouseReleaseEvent(QMouseEvent*)成員函數
④QPushButton調用click()成員函數
⑤觸發信號SIGNAL(clicked())
(5)實例
實例一:自定義事件處理函數
鼠標左鍵右鍵
wigth.h
<span style="color:#000000"><code class="language-cpp"><span style="color:#f1fa8c">#<span style="color:#ff79c6">ifndef</span> WIDGET_H</span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">define</span> WIDGET_H</span> ? <span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><QWidget></span></span> ? <span style="color:#ff79c6">class</span> Widget <span style="color:#6272a4">:</span> <span style="color:#ff79c6">public</span> QWidget <span style="color:#999999">{</span>Q_OBJECT <span style="color:#ff79c6">protected</span><span style="color:#6272a4">:</span><span style="color:#ff79c6">void</span> <span style="color:#8be9fd">mousePressEvent</span><span style="color:#999999">(</span>QMouseEvent <span style="color:#6272a4">*</span>event<span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#ff79c6">public</span><span style="color:#6272a4">:</span><span style="color:#8be9fd">Widget</span><span style="color:#999999">(</span>QWidget <span style="color:#6272a4">*</span>parent <span style="color:#6272a4">=</span> <span style="color:#f1fa8c">0</span><span style="color:#999999">)</span><span style="color:#999999">;</span> ?<span style="color:#6272a4">~</span><span style="color:#8be9fd">Widget</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#999999">}</span><span style="color:#999999">;</span> ? <span style="color:#f1fa8c">#<span style="color:#ff79c6">endif</span> </span><span style="color:#6272a4">// WIDGET_H</span> ? </code></span>widgth.cpp
<span style="color:#000000"><code class="language-cpp"><span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c">"widget.h"</span></span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><QMouseEvent></span></span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><QDebug></span></span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><QMenu></span></span> Widget<span style="color:#6272a4">::</span><span style="color:#8be9fd">Widget</span><span style="color:#999999">(</span>QWidget <span style="color:#6272a4">*</span>parent<span style="color:#999999">)</span><span style="color:#6272a4">:</span> <span style="color:#8be9fd">QWidget</span><span style="color:#999999">(</span>parent<span style="color:#999999">)</span> <span style="color:#999999">{</span> <span style="color:#999999">}</span> ? <span style="color:#ff79c6">void</span> Widget<span style="color:#6272a4">::</span><span style="color:#8be9fd">mousePressEvent</span><span style="color:#999999">(</span>QMouseEvent <span style="color:#6272a4">*</span>event<span style="color:#999999">)</span> <span style="color:#999999">{</span><span style="color:#ff79c6">if</span><span style="color:#999999">(</span>event<span style="color:#6272a4">-</span><span style="color:#6272a4">></span><span style="color:#8be9fd">button</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#6272a4">==</span>Qt<span style="color:#6272a4">::</span>LeftButton<span style="color:#999999">)</span><span style="color:#999999">{</span><span style="color:#8be9fd">qDebug</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#6272a4"><<</span><span style="color:#f1fa8c">"LeftButton clicked!"</span><span style="color:#999999">;</span><span style="color:#999999">}</span><span style="color:#ff79c6">else</span> <span style="color:#ff79c6">if</span><span style="color:#999999">(</span>event<span style="color:#6272a4">-</span><span style="color:#6272a4">></span><span style="color:#8be9fd">button</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#6272a4">==</span>Qt<span style="color:#6272a4">::</span>RightButton<span style="color:#999999">)</span><span style="color:#999999">{</span><span style="color:#8be9fd">qDebug</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#6272a4"><<</span><span style="color:#f1fa8c">"RightButton clicked!"</span><span style="color:#999999">;</span><span style="color:#999999">}</span> ? <span style="color:#999999">}</span> ? Widget<span style="color:#6272a4">::</span><span style="color:#6272a4">~</span><span style="color:#8be9fd">Widget</span><span style="color:#999999">(</span><span style="color:#999999">)</span> <span style="color:#999999">{</span><span style="color:#999999">}</span> ? </code></span>main.cpp
<span style="color:#000000"><code class="language-cpp"><span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><QApplication></span></span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c">"widget.h"</span></span> ? <span style="color:#ff79c6">int</span> <span style="color:#8be9fd">main</span><span style="color:#999999">(</span><span style="color:#ff79c6">int</span> argc<span style="color:#999999">,</span> <span style="color:#ff79c6">char</span> <span style="color:#6272a4">*</span>argv<span style="color:#999999">[</span><span style="color:#999999">]</span><span style="color:#999999">)</span> <span style="color:#999999">{</span>QApplication <span style="color:#8be9fd">a</span><span style="color:#999999">(</span>argc<span style="color:#999999">,</span> argv<span style="color:#999999">)</span><span style="color:#999999">;</span>Widget w<span style="color:#999999">;</span>w<span style="color:#999999">.</span><span style="color:#8be9fd">show</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">;</span><span style="color:#ff79c6">return</span> a<span style="color:#999999">.</span><span style="color:#8be9fd">exec</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#999999">}</span> ? </code></span>實例二:自定義事件處理函數
QMyPushButton.h
<span style="color:#000000"><code class="language-cpp"><span style="color:#f1fa8c">#<span style="color:#ff79c6">ifndef</span> _QMYPUSHBUTTON_H_</span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">define</span> _QMYPUSHBUTTON_H_</span><span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><QPushButton></span></span><span style="color:#ff79c6">typedef</span> <span style="color:#ff79c6">void</span> <span style="color:#999999">(</span>QButtonListener<span style="color:#999999">)</span><span style="color:#999999">(</span>QObject<span style="color:#6272a4">*</span><span style="color:#999999">,</span>QMouseEvent<span style="color:#6272a4">*</span><span style="color:#999999">)</span><span style="color:#999999">;</span><span style="color:#ff79c6">class</span> QMyPushButton <span style="color:#6272a4">:</span> <span style="color:#ff79c6">public</span> QPushButton <span style="color:#999999">{</span>Q_OBJECT<span style="color:#ff79c6">protected</span><span style="color:#6272a4">:</span>QButtonListener<span style="color:#6272a4">*</span> m_listener<span style="color:#999999">;</span><span style="color:#6272a4">//重寫QPushButton的事件處理函數 就有可能不會產生clicked信號</span><span style="color:#ff79c6">void</span> <span style="color:#8be9fd">mouseReleaseEvent</span><span style="color:#999999">(</span>QMouseEvent <span style="color:#6272a4">*</span>e<span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#ff79c6">public</span><span style="color:#6272a4">:</span><span style="color:#ff79c6">explicit</span> <span style="color:#8be9fd">QMyPushButton</span><span style="color:#999999">(</span>QWidget<span style="color:#6272a4">*</span> parent <span style="color:#6272a4">=</span> <span style="color:#f1fa8c">0</span><span style="color:#999999">,</span> QButtonListener<span style="color:#6272a4">*</span> listener <span style="color:#6272a4">=</span> <span style="color:#f1fa8c">0</span><span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#999999">}</span><span style="color:#999999">;</span><span style="color:#f1fa8c">#<span style="color:#ff79c6">endif</span> </span><span style="color:#6272a4">// _QMYPUSHBUTTON_H_</span> </code></span>QMyPushButton.cpp
<span style="color:#000000"><code class="language-cpp"><span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c">"QMyPushButton.h"</span></span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><QMouseEvent></span></span>QMyPushButton<span style="color:#6272a4">::</span><span style="color:#8be9fd">QMyPushButton</span><span style="color:#999999">(</span>QWidget<span style="color:#6272a4">*</span> parent<span style="color:#999999">,</span> QButtonListener<span style="color:#6272a4">*</span> listener<span style="color:#999999">)</span><span style="color:#6272a4">:</span><span style="color:#8be9fd">QPushButton</span><span style="color:#999999">(</span>parent<span style="color:#999999">)</span> <span style="color:#999999">{</span>m_listener <span style="color:#6272a4">=</span> listener<span style="color:#999999">;</span> <span style="color:#999999">}</span><span style="color:#6272a4">//重寫改寫事件處理函數,會改變程序的行為。</span> <span style="color:#ff79c6">void</span> QMyPushButton<span style="color:#6272a4">::</span><span style="color:#8be9fd">mouseReleaseEvent</span><span style="color:#999999">(</span>QMouseEvent <span style="color:#6272a4">*</span>e<span style="color:#999999">)</span> <span style="color:#999999">{</span><span style="color:#ff79c6">if</span><span style="color:#999999">(</span>m_listener <span style="color:#6272a4">!=</span> <span style="color:#f1fa8c">NULL</span><span style="color:#999999">)</span><span style="color:#999999">{</span><span style="color:#6272a4">//調用自定義的事件處理函數,盡管按鈕的clicked信號被連接到onMyButtonClicked槽函數,</span><span style="color:#6272a4">//但因自定義的m_listener函數里并不觸發clicked信號,從而槽函數不會被調用。</span><span style="color:#8be9fd">m_listener</span><span style="color:#999999">(</span><span style="color:#ff79c6">this</span><span style="color:#999999">,</span> e<span style="color:#999999">)</span><span style="color:#999999">;</span>e<span style="color:#6272a4">-</span><span style="color:#6272a4">></span><span style="color:#8be9fd">accept</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">;</span><span style="color:#6272a4">//事件被接收,就不再傳遞到父QWidget</span><span style="color:#8be9fd">setDown</span><span style="color:#999999">(</span><span style="color:#8be9fd">false</span><span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#6272a4">//按鈕設置為“彈起”狀態</span><span style="color:#999999">}</span><span style="color:#ff79c6">else</span><span style="color:#999999">{</span><span style="color:#6272a4">//父類的mouseReleaseEvent會去調用clicked(),并觸發SIGNAL(clicked())</span><span style="color:#6272a4">//從而調用到連接到該信號的槽函數(本例為onMyButtonClicked())</span>QPushButton<span style="color:#6272a4">::</span><span style="color:#8be9fd">mouseReleaseEvent</span><span style="color:#999999">(</span>e<span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#6272a4">//調用父類</span><span style="color:#999999">}</span> <span style="color:#999999">}</span> </code></span>Widget.h
<span style="color:#000000"><code class="language-cpp"><span style="color:#f1fa8c">#<span style="color:#ff79c6">ifndef</span> _WIDGET_H_</span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">define</span> _WIDGET_H_</span><span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><QWidget></span></span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c">"QMyPushButton.h"</span></span><span style="color:#ff79c6">class</span> Widget <span style="color:#6272a4">:</span> <span style="color:#ff79c6">public</span> QWidget <span style="color:#999999">{</span>Q_OBJECTQMyPushButton myButton<span style="color:#999999">;</span><span style="color:#ff79c6">protected</span> slots<span style="color:#6272a4">:</span><span style="color:#ff79c6">void</span> <span style="color:#8be9fd">onMyButtonClicked</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">;</span><span style="color:#ff79c6">public</span><span style="color:#6272a4">:</span><span style="color:#8be9fd">Widget</span><span style="color:#999999">(</span>QWidget <span style="color:#6272a4">*</span>parent <span style="color:#6272a4">=</span> <span style="color:#f1fa8c">0</span><span style="color:#999999">)</span><span style="color:#999999">;</span><span style="color:#6272a4">~</span><span style="color:#8be9fd">Widget</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#999999">}</span><span style="color:#999999">;</span><span style="color:#f1fa8c">#<span style="color:#ff79c6">endif</span> </span><span style="color:#6272a4">// _WIDGET_H_</span> </code></span>Widget.cpp
<span style="color:#000000"><code class="language-cpp"><span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c">"Widget.h"</span></span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><qDebug></span></span><span style="color:#6272a4">//自定義事件處理函數</span> <span style="color:#ff79c6">void</span> <span style="color:#8be9fd">onMyButtonMouseRelease</span><span style="color:#999999">(</span>QObject<span style="color:#6272a4">*</span> sender<span style="color:#999999">,</span> QMouseEvent<span style="color:#6272a4">*</span> e<span style="color:#999999">)</span> <span style="color:#999999">{</span><span style="color:#8be9fd">qDebug</span><span style="color:#999999">(</span><span style="color:#999999">)</span> <span style="color:#6272a4"><<</span> <span style="color:#f1fa8c">"onMyButtonMouseRelease(QObject* sender, QMouseEvent* e)"</span><span style="color:#999999">;</span> <span style="color:#999999">}</span>Widget<span style="color:#6272a4">::</span><span style="color:#8be9fd">Widget</span><span style="color:#999999">(</span>QWidget <span style="color:#6272a4">*</span>parent<span style="color:#999999">)</span><span style="color:#6272a4">:</span> <span style="color:#8be9fd">QWidget</span><span style="color:#999999">(</span>parent<span style="color:#999999">)</span><span style="color:#999999">,</span><span style="color:#8be9fd">myButton</span><span style="color:#999999">(</span><span style="color:#ff79c6">this</span><span style="color:#999999">,</span> onMyButtonMouseRelease<span style="color:#999999">)</span> <span style="color:#6272a4">//實驗2:myButton(this, 0)</span> <span style="color:#999999">{</span>myButton<span style="color:#999999">.</span><span style="color:#8be9fd">setText</span><span style="color:#999999">(</span><span style="color:#f1fa8c">"QMyPushButton"</span><span style="color:#999999">)</span><span style="color:#999999">;</span><span style="color:#8be9fd">connect</span><span style="color:#999999">(</span><span style="color:#6272a4">&</span>myButton<span style="color:#999999">,</span> <span style="color:#8be9fd">SIGNAL</span><span style="color:#999999">(</span><span style="color:#8be9fd">clicked</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">)</span><span style="color:#999999">,</span> <span style="color:#ff79c6">this</span><span style="color:#999999">,</span> <span style="color:#8be9fd">SLOT</span><span style="color:#999999">(</span><span style="color:#8be9fd">onMyButtonClicked</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">)</span><span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#999999">}</span><span style="color:#6272a4">//槽函數,用于接收按鈕的clicked信號</span> <span style="color:#ff79c6">void</span> Widget<span style="color:#6272a4">::</span><span style="color:#8be9fd">onMyButtonClicked</span><span style="color:#999999">(</span><span style="color:#999999">)</span> <span style="color:#999999">{</span><span style="color:#8be9fd">qDebug</span><span style="color:#999999">(</span><span style="color:#999999">)</span> <span style="color:#6272a4"><<</span> <span style="color:#f1fa8c">"onMyButtonClicked()"</span> <span style="color:#999999">;</span> <span style="color:#999999">}</span>Widget<span style="color:#6272a4">::</span><span style="color:#6272a4">~</span><span style="color:#8be9fd">Widget</span><span style="color:#999999">(</span><span style="color:#999999">)</span> <span style="color:#999999">{</span> <span style="color:#999999">}</span> </code></span>main.cpp
<span style="color:#000000"><code class="language-cpp"><span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c">"Widget.h"</span></span> <span style="color:#f1fa8c">#<span style="color:#ff79c6">include</span> <span style="color:#f1fa8c"><QApplication></span></span><span style="color:#ff79c6">int</span> <span style="color:#8be9fd">main</span><span style="color:#999999">(</span><span style="color:#ff79c6">int</span> argc<span style="color:#999999">,</span> <span style="color:#ff79c6">char</span> <span style="color:#6272a4">*</span>argv<span style="color:#999999">[</span><span style="color:#999999">]</span><span style="color:#999999">)</span> <span style="color:#999999">{</span>QApplication <span style="color:#8be9fd">a</span><span style="color:#999999">(</span>argc<span style="color:#999999">,</span> argv<span style="color:#999999">)</span><span style="color:#999999">;</span>Widget w<span style="color:#999999">;</span>w<span style="color:#999999">.</span><span style="color:#8be9fd">show</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">;</span><span style="color:#ff79c6">return</span> a<span style="color:#999999">.</span><span style="color:#8be9fd">exec</span><span style="color:#999999">(</span><span style="color:#999999">)</span><span style="color:#999999">;</span> <span style="color:#999999">}</span> </code></span>(6)事件(QEvent)和信號(SIGNAL)的不同
例如:單擊界面上的按鈕,那么就會產生鼠標事件QMouseEvent(不是按鈕產生的),由于按鈕被按下了,所以他會發出一個單擊信號clicked()信號(是按鈕產生的),這里只考慮單擊信號而不用考慮鼠標事件,但如果要設計一個按鈕,或者當單擊按鈕時讓它產生別的效果,此時就要考慮鼠標事件了,由此,信號和事件是兩個不同層面的東西,發出者不同,作用不同。Qt中,所有的QObject的子類實例均可對事件接收和處理!
3. 小結
(1)Qt中的事件和信號不同
(2)事件由QObject對象進行處理
(3)信號由QObject對象觸發
(4)重寫事件處理函數可能改變程序行為
(5)信號的觸發不會對程序行為造成影響
(6)事件處理是在實際工程開發中應用非常普遍的
總結
以上是生活随笔為你收集整理的Qt 之 消息机制和事件讲解的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 解决:Error response fr
- 下一篇: 解决:No goals have bee