Qt:在默认浏览器中打开目标链接,不会泄漏内存

通过互联网搜索,我遇到了很多方法,大部分是非功能性的,非特定性的或部分功能性的,用QWebView和打开URL来做各种事情。

经过大量的咒骂和诅咒之后,我设法得到了一个例子来做我想做的事情,这是正常打开的正常链接,并打开任何在外部浏览器中请求新窗口的内容; 然而,这有一个困难。 它泄漏内存,因为我制作了一堆额外的WebViews ,直到进程退出时才清理干净。 我怎样才能做到这一点,而不会泄漏内存?

请原谅我对Qt的相当高雅的理解。 在这一点上我只用了几个小时。

SSCCE:

test.hpp

#include <QMainWindow>
#include <QWebView>

class Window : public QMainWindow {
  Q_OBJECT

public:
  Window();

private:
  QWebView* m_web;

private slots:
};

class WebPage : public QWebPage {
  Q_OBJECT

public:
    bool acceptNavigationRequest(QWebFrame *frame, const QNetworkRequest &request, NavigationType type);
};

class WebView : public QWebView {
  Q_OBJECT

public:
  QWebView* createWindow(QWebPage::WebWindowType type);
};

TEST.CPP

#include <QApplication>
#include <QGridLayout>
#include <QNetworkRequest>
#include <QDesktopServices>

#include "test.hpp"

Window::Window() :
    QMainWindow() {
  m_web = new WebView;
  m_web->setHtml("<div align="center"><a href="http://www.google.com/">Same Window</a> <a href="http://www.google.com/" target="_blank">New Window</a></div>");

  setCentralWidget(m_web);
}

bool WebPage::acceptNavigationRequest(QWebFrame*, QNetworkRequest const& request, NavigationType) {
  QDesktopServices::openUrl(request.url());
  return false;
}

QWebView* WebView::createWindow(QWebPage::WebWindowType) {
  auto res = new WebView;
  auto page = new WebPage;
  res->setPage(page);
  return res;
}

int main(int argc, char *argv[]) {
  QApplication a(argc, argv);

  Window w;
  w.show();

  return a.exec();
}

test.pro

QT += core gui network webkitwidgets widgets

TEMPLATE = app
TARGET = test
INCLUDEPATH += .

CONFIG += c++11

# Input
SOURCES += test.cpp
HEADERS += test.hpp

编译和运行

qmake test.pro
make
./test

看起来,该视图在extern浏览器中呈现页面后变得毫无用处。 您可能只需使用deleteLater()将ExternalWebView安排为删除操作即可:

#include <iostream>

#include <QApplication>
#include <QGridLayout>
#include <QNetworkRequest>
#include <QDesktopServices>

#include <QEvent>
#include <QMainWindow>
#include <QWebView>

class ExternWebPage : public QWebPage {
    //Q_OBJECT

    public:
    ExternWebPage(QObject* parent = 0)
    :   QWebPage(parent)
    {
        std::cout << "ExternWebPage" << std::endl;
    }

    ~ExternWebPage() {
        std::cout << "Destroy ExternWebPage" << std::endl;
    }

    virtual bool event(QEvent *e) {
        static unsigned counter;
        std::cout << ++counter << " ExternWebPage: " << e->type() << std::endl;
        return QWebPage::event(e);
    }

    bool acceptNavigationRequest(QWebFrame *, const QNetworkRequest &request, NavigationType) {
        QDesktopServices::openUrl(request.url());
        return false;
    }
};


class ExternWebView : public QWebView {
    //Q_OBJECT

    public:
    ExternWebView(QWidget* parent = 0)
    :   QWebView(parent)
    {
        std::cout << "ExternWebView" << std::endl;
    }
    ~ExternWebView() { std::cout << "Destroy ExternWebView" << std::endl; }
    virtual bool event(QEvent *e) {
        static unsigned counter;
        std::cout << ++counter << " ExternWebView: " << e->type() << std::endl;
        return QWebView::event(e);
    }
};


class InternalWebView : public QWebView {
    //Q_OBJECT

    public:
    InternalWebView(QWidget* parent = 0)
    :   QWebView(parent)
    {}
    QWebView* createWindow(QWebPage::WebWindowType) {
        auto res = new ExternWebView();
        res->setPage(new ExternWebPage(res));
        res->deleteLater();
        return res;
    }
};


class Window : public QMainWindow {
    //Q_OBJECT

    public:
    Window()
    :   QMainWindow()
    {
        std::cout << "Window" << std::endl;
        auto web = new InternalWebView(this);
        web->setHtml("<div align="center"><a href="http://www.google.com/">Same Window</a> <a href="http://www.google.com/" target="_blank">New Window</a></div>");
        setCentralWidget(web);
    }
    ~Window() { std::cout << "Destroy Window" << std::endl; }

};


int main(int argc, char *argv[]) {
  QApplication a(argc, argv);
  Window w;
  w.show();
  return a.exec();
}

没有删除的测试:

Window
ExternWebView
1 ExternWebView: 68
ExternWebPage
2 ExternWebView: 74
3 ExternWebView: 75
1 ExternWebPage: 43
2 ExternWebPage: 43
3 ExternWebPage: 43
4 ExternWebPage: 43
5 ExternWebPage: 43
Destroy Window

稍后删除测试:

ExternWebView
1 ExternWebView: 68
ExternWebPage
2 ExternWebView: 74
3 ExternWebView: 75
4 ExternWebView: 52
Destroy ExternWebView
Destroy ExternWebPage
Destroy Window

QDesktopServices::openUrl(QUrl("http://stackoverflow.com/"));

由于这是内置函数,因此不应该有任何内存泄漏。


程序中没有泄漏,因为qt内存管理系统将处理在堆上创建的对象。

首先,setPage将使您的视图对象成为页面对象的父对象。 这意味着当视图对象被销毁时页面对象将被删除。

其次,由于视图没有父母,所以你总是会得到一个窗口。 当你关闭窗口或者结束程序时它会被释放。 这就是为什么我说这将会被qt的内存管理系统所关注。

现在,当您使用内存分析器程序(如valgrind)运行程序时,可能会发生泄漏,这可能是或可能不是真正的泄漏。 你需要识别它们,并将其过滤掉。

链接地址: http://www.djcxy.com/p/75933.html

上一篇: Qt: Open links with target in default browser, without leaking memory

下一篇: Golang 1.2: Unzip password protected zip file?