通过互联网搜索,我遇到了很多方法,大多数是非功能性的,非特定的或部分功能性的,可以使用QWebView
和打开网址来执行各种操作。
经过多次咒骂和诅咒之后,我成功地做了一个例子来做我想要的事情,这通常是开放的普通链接,并打开任何在外部浏览器中请求新窗口的内容;然而,有一个障碍。它泄漏了内存,因为我制作了一堆额外的WebViews
,直到进程退出时才会被清除。如何在不泄漏记忆的情况下做到这一点?
请提前原谅我对Qt的二年级理解。我此时只用了几个小时。
SSCCE:
#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);
};
#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();
}
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
答案 0 :(得分:4)
在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
答案 1 :(得分:1)
QDesktopServices::openUrl(QUrl("http://stackoverflow.com/"));
由于这是内置函数,因此不应有任何内存泄漏。
答案 2 :(得分:0)
程序中没有泄漏,因为qt内存管理系统将负责处理堆上创建的对象。
首先,setPage将使您的视图对象成为页面对象的父对象。这意味着当视图对象被破坏时,页面对象将被删除。
其次,由于视图没有父级,因此您总是会得到一个窗口。它将在您关闭窗口或结束程序时释放。这就是为什么我说它将被qt的内存管理系统所关注。
现在,当您使用内存分析程序(如valgrind)运行程序时,可能会出现泄漏,这可能是也可能不是真正的泄漏。您需要识别它们并将其过滤掉。