首頁 > 軟體

QT中視窗關閉自動銷燬的實現範例

2022-05-16 13:00:06

我們知道c++記憶體管理的一個規則:new出來的物件,一定要delete。

我們實現一個彈窗的時候,有時候繼承widget,有時候繼承dialog。不管哪種,表單呼叫show方法後,不可能直接delete。

當然,new該表單的時候,指定了父物件,就可以不用顯示呼叫delete,它會隨著父物件的銷燬而銷燬。

那這時候,關閉彈窗的操作,就僅僅是隱藏了。為了實現真正的delete。我們需要設施視窗的一個屬性 DeleteOnClose,具體方法如下:

setAttribute(Qt::WA_DeleteOnClose, true);

QT 視窗建立時,無論是模態或者是非模態的,可以通過設定表單屬性來進行銷燬設定,設定該屬性後,表單如果關閉,表單申請的記憶體會立即銷燬。範例:

    QWidget *qui = new QWidget;
    qui->setTitle("新表單測試");
    qui->setAlignment(Qt::AlignCenter);
    qui->setAttribute(Qt::WA_DeleteOnClose, true);
    qui->show();

表單gui關閉後,gui所佔用的記憶體會進行釋放。

qt關於視窗關閉觸發函數/訊號

方法一、

視窗右上角的X按鍵會導致其在不給出任何提示的情況下直接退出,
當點選右上角的x按鍵時,會觸發Qt中的一個事件處理常式:void QWidget::closeEvent ( QCloseEvent * e ),預設情況該函數無任何提示性訊息,如果需提示則需要重寫該函數

void Dialog::closeEvent ( QCloseEvent * e )
{
    if( QMessageBox::question(this,
                             tr("Quit"),
                             tr("Are you sure to quit this application?"),
                              QMessageBox::Yes, QMessageBox::No )
                   == QMessageBox::Yes){
        e->accept();//不會將事件傳遞給元件的父元件

        qDebug()<<"ok";
    }
    else
      e->ignore();  
}

說明:

事件的ignore函數表示忽略事件,將其傳到父元件進行處理
事件的accept函數表示接受事件,元件自己進行處理,不會將事件傳遞給父元件
但是在這個closeEvent函數中,這兩個函數尤其不一樣的理解:對於視窗關閉QCloseEvent事件,
呼叫accept()意味著 Qt 會停止事件的傳播,視窗關閉;
呼叫ignore()則意味著事件繼續傳播,即阻止視窗關閉。
此外如果需要直接退出某個應用程式,可以直接呼叫Qt中的一個全域性指標:qApp ,全域性指標qApp指向全域性的QApplication物件。

//在需要停止程式執行的位置直接加入如下程式碼
#include <QtGui/QApplication>
...
  qApp->quit();

說明:

qApp是一個全域性的函數,可以在Qt說明檔案中直接找到,不是main函數定義的QApplication物件(QApplication物件也可以退出,使用 app->exit(0);或者 app->quit();具體可以參考【1】)
qApp->quit()關閉的是整個應用程式,不只是視窗,
一般情況下關閉視窗應用程式會直接退出,如果要求關閉視窗時,應用程式不退出,則需要設定QApplication的屬性:
QApplication::setQuitOnLastWindowClosed(false);
 

方法二、

QWidget的close槽函數是像widget傳送QCloseEvent,如果widget未設定Qt::WA_DeleteOnClose標誌的話,將隱藏widget,並不會銷燬相關資源。如果設定了該標誌,那麼會再傳送destroy訊號,銷燬相關資源。

(多說一句:QWindow的close槽是呼叫destroy來銷燬視窗資源的。不過我們一般不直接使用QWindow。)

Qt幫助檔案中的介紹:

呼叫close()方法後首先它會向widget傳送一個關閉事件(QCloseEvent)。如果widget接受了關閉事件(QCloseEvent),視窗將會隱藏(實際上呼叫hide())。
如果widget不接受關閉事件,那麼視窗將什麼也不做。預設情況下widget會接受關閉事件,我們可以重寫QCloseEvent事件,可以選擇接受或者不接受。

如果widget設定了Qt::WA_DeleteOnClose屬性,widget將會被釋放。不管widget是否可見,關閉事件都會傳遞給widget。即接收到QCloseEvent事件後,
除了呼叫hide()方法將視窗隱藏,同時會呼叫deleteLater()方法將視窗釋放掉,不會再佔用資源。

所以說呼叫close()並不一定就會將視窗物件銷燬。而只有設定了 Qt::WA_DeleteOnClose屬性才會刪除銷燬。如果這個屬性沒有設定,close()的作用和hide(),
setvisible(false)一樣,只會隱藏視窗物件而已,並不會銷燬該物件。

這樣的話我們可以給QLabel(拿它當一個例子)設定Qt::WA_DeleteOnClose

QLabel *label = new QLabel;
label->setAttribute(Qt::WA_DeleteOnClose);

之後用connect使用label的destroyed訊號繫結一個槽函數即可

connect(label, &QLabel::destroyed, this, [=]() {
    QString file_path = QFileDialog::getSaveFileName(this, "save file", "tmp.png", "(*.png);;all files(*.*)");
        if(!file_path.isEmpty())
        ......
});

到此這篇關於QT中視窗關閉自動銷燬的實現範例的文章就介紹到這了,更多相關QT 視窗關閉自動銷燬內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com