【golang-GUI開發】qt之signal和slot(一)

来源:https://www.cnblogs.com/apocelipes/archive/2018/07/15/9315166.html
-Advertisement-
Play Games

想了很久,我決定還是先從signal和slot(信號槽)開始講起。 signal和slot大家一定不陌生,先看一段示例(選自文檔): 使用signal和slot的類必須包含Q_OBJECT巨集,聲明slot需要使用public/private/protected slots:,signal則需要sig ...


想了很久,我決定還是先從signal和slot(信號槽)開始講起。

signal和slot大家一定不陌生,先看一段示例(選自文檔):

 1 class Counter : public QObject
 2 {
 3     Q_OBJECT
 4 
 5 public:
 6     Counter() { m_value = 0; }
 7 
 8     int value() const { return m_value; }
 9 
10 public slots:
11     void setValue(int value);
12 
13 signals:
14     void valueChanged(int newValue);
15 
16 private:
17     int m_value;
18 };

使用signal和slot的類必須包含Q_OBJECT巨集,聲明slot需要使用public/private/protected slots:,signal則需要signals:。

這些其實都是巨集,它們會指示moc做相應的代碼生成,這樣Qt程式才可以發送信號,並讓slot與signal相連接。

可是golang並沒有巨集,那麼在qt里我們要怎麼做呢?

信號----Signal

1. 信號----signal的定義

想要自定義signals,我們需要用到golang的一個簡單特性--struct tags。

tags被廣泛的用於golang的世界里,從標準庫encoding/json到廣泛使用的orm(xorm,gorm),tags的身影無處不在。tags之所以應用廣泛是因為它可以被reflect取到,

依賴於強大的reflect包,可以通過tags來實現各種各樣的功能,其中就包括Qt的moc擴展。

下麵我們看一下一個帶有兩個自定義signal的自定義組件:

1 import "github.com/therecipe/qt/core"
2 
3 type MyWidget struct {
4     core.QObject
5     
6     _ func() `signal:"dataChanged"`
7     _ func(int) `signal:"valueChanged"`
8 }

首先看到第四行,

core.QObject

所有需要自定義slot和signal的類都必須是core.QObject的派生類型,如果不是直接繼承自QObject,那麼直接繼承的類型必須要直接或間接的繼承自QObject。

同時要註意,不要用*core.QObject的形式,這會導致qtmoc忽略這個類,最終不能處理moc擴展引發問題

_ func() `signal:"dataChanged"`
_ func(int) `signal:"valueChanged"`

我們定義了兩個signal,第一個不帶任何參數,第二個帶有一個int類型參數。

qtmoc會把tags的內容用strings.Title做處理,也就是說dataChanged會變成DataChanged,這就是我們定義的信號的名字。

接著qtmoc會根據這個名字以及tags所在成員的類型生成自定義控制項類的三個成員方法:Connect[signal name],Disconnect[signal name],[signal name],

在本例中就是:ConnectDataChanged,DisconnectDataChanged和Datachanged。

 

2. 信號----signal的連接

想要和signal連接,需要用到前面提到的Connect[signal name]函數。

qtmoc會根據signal的類型來生成Connect函數,這裡的ConnectDataChanged的原型就是func ConnectDataChanged( f func() )。

需要連接這個signal時,調用它並把signal處理函數傳遞為參數即可

func sample() {
    fmt.Println("Data has been changed.")
}

widget := NewMyWidget(nil) // 這裡是創建我們的自定義組件,後面的文章我們會重點講解
// Qt5中與signal相連的可以是任何函數,在qt里也是一樣,所以我們用一個外部函數來處理signal,在實際開發中還是推薦用類的成員方法或者slot進行處理
widget.ConnectDataChanged(sample)

這裡我們把sample和信號DataChanged相連,每次觸發這個信號時都會列印出“Data has been changed.”這句信息。

如果想要取消和某個信號的連接,需要使用Disconnect[signal name]函數,它不帶參數,調用它意味著取消signal與上一次使用Connect[signal name]時作為參數的函數的連接。

widget.DisconnectDataChanged()
// 我們取消了sample函數與DataChanged的連接

 

3. 信號----signal的觸發

在C++中要觸發一個信號,只需要如下代碼:

emit DataChanged()
emit ValueChanged(value)

emit?在C++和golang里都沒見過的語法。。。。。。沒錯,這也是Qt的moc擴展。

還記得我們說道qtmoc會根據signal tags生成三個成員方法嗎,ConnectDataChanged,DisconnectDataChanged,DataChanged

第三個函數就是我們用來觸發信號的。

信號觸發函數用來代替emit,它自身是一個根據signal tags前的類型生成的函數,所以MyWidget.DataChanged的類型是func f();而ValueChanged函數的類型就是func f(value int)。

註意,與Qt一樣,signal不可以擁有返回值

下麵是觸發信號的示例:

// 觸發DataChanged信號
widget.DataChanged()

value := 5
// 觸發ValueChanged信號並傳遞參數
widget.ValueChanged(1)
widget.ValueChanged(100)
widget.ValueChanged(value)

觸發信號之後,之前與之相連的函數就會被調用了。

 

4. 信號----signal的自動連接

如果自定義的signal比較多,那麼一個個的調用Connect[signal name]不僅麻煩低效,還會帶來維護上的困難,所以qt提供了自動連接的功能。

先看代碼:

import (
    "github,com/therecipe/qt/widgets"
)

type Auto struct {
    widgets.QLabel

    _ func() `signal:"dataChanged,auto"`
    _ func(string) `signal:"valueChanged,auto(this.QLabel.SetText)"`
}

我們看到在signal的名字後面多了一個auto。

這個auto是告訴qtmoc這個信號需要connect一個和signal tags里名字相同的成員方法,在這裡成員函數的名字必須和tags里的相同,而不是經過strings.Title處理過的signal名字。

然後我們定義並實現這個和DataChanged連接的成員函數:

func (a *Auto) dataChanged() {
    fmt.Println("Data has been changed.")
}

這樣你無需再顯示調用ConnectDataChanged,DataChanged將自動和成員函數dataChanged連接。

我們還看到有auto(this.QLabel.SetText)的寫法,這是在自定義類型繼承自其他QObject及其派生類時,可以自動連接基類的成員方法。

this是指當前的對象;

QLabel或是其他類型名錶示繼承的基類;

SetText是基類的成員函數,它將與DataChanged信號相關聯。這裡寫成setText也可以,因為在()里的函數名會被strings.Title處理。

每當我們觸發信號時:

widget := NewAuto(nil, 0)

widget.DataChanged()
widget.ValueChanged("signal & slot")

相應的成員函數就會被調用,上面的代碼會有如下反應:

// 因為DataChanged信號而被觸發
widget.dataChanged()

// 因為ValueChanged信號而被觸發
widget.QLabel.SetText("signal & slot")
// 等價於
widget.SetText("signal & slot")

有人會問,那可不可以用`signal:"dataChanged,auto(this.Myfunc)"`自己指定想要和信號connect的函數呢?

答案是暫時不可以。目前auto只有上面兩種用法,不過作者以及把實現自動連接成員變數的成員函數和自定義連接函數加入了開發計劃中,相信不久之後就能用上這些功能了。

 

以上就是qt中signal的具體用法,下一篇我們將會介紹slot的詳細用法。

如有疑問歡迎在評論中提出。

參考:

http://doc.qt.io/qt-5/signalsandslots.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 有時候出現這種怪異的問題,是由於多個版本的class存在。 比如說:某個java編譯成class後,放到classes下麵,然後lib目錄下,也有這個class所在的jar包,這樣就導致classpath實際上有兩個相同的class(以上來自百度)解決辦法:把項目的編譯文件刪除,知後重新編譯(如果找 ...
  • 介面是一個或多個方法簽名名的集合,定義方式如下 `type Interface_Name interface { method_a() string method_b() int .... } ` 只要某個類型擁有該介面的所有方法簽名,就算實現該介面,無需顯示聲明實現了那個介面,這稱為structu ...
  • 一、簡介 JDK1.5之前都是通過synchronized關鍵字實現併發同步,而JDK1.5以後Doug Lea大師開發了current包下的類,通過JAVA代碼實現了synchronized關鍵的語義。然而在current包下的這些類的實現大部分都不離不開一個基礎組件 AQS(AbstractQu ...
  • html的每一個標簽都共有的屬性有'id','class','style' ...
  • 一、Hbernate中的日誌框架_整合log4j(瞭解) Hibernate 利用 Simple Logging Facade for Java (SLF4J)來記錄不同系統事件的日誌。SLF4J 可以根據你選擇的綁定把日誌輸出到幾個日誌框架(NOP、Simple、log4j version1.2、 ...
  • 問題描述: 在activemq的監聽器中,通過註解@Autowired或@Resource註入bean時,獲取到的bean為null。調用該bean的方法時會報空指針異常。 問題原因: 當調用bean內方法時,spring容器中還沒有完成對註解bean的掃描,dispatcher.xml中配置的註解 ...
  • 實現對DataFrame對象隨機採樣 pandas是基於numpy建立起來的,所以numpy大部分函數可作用於DataFrame和Series數據結構。 numpy.random.permutation(n)函數可以產生0~n範圍內的n個隨機數,輸出形式為numpy數組。 In: Out: 新建一個 ...
  • 1、java中的scanner: scanner 是java中用來進行人機交互的函數; 使用前需要導入:java.util.Scanner包 import java.util.Scanner; 在使用時有以下語法: Scanner sc=new Scanner(System.in);//建立一個鍵盤 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...