目标:在 Windows 7、10 下的 Qt5 C++ 应用程序中拥有一个可滚动的自定义小部件列表,数量达数千(可能更多)。
问题:将窗口最小化到任务栏并再次恢复后,程序停止响应。虽然它不会崩溃。 CPU 使用率常量为 25%。即使在等待几分钟后,GUI 也不会再次响应。此外,通常会消耗大量内存(超过 200M),我认为即使对于 100k QLabels(每个 QLabel 大约 2k)来说也太多了。
Here是针对类似问题的一些建议解决方案,但我认为这些解决方案不适合我的情况。
示例: 下面的示例说明了这个问题。为了演示,使用了 QLabels 列表,但它可以是派生自 QWidget 的任何类。
主窗口.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QMainWindow>
#include <QScrollArea>
#include <QVBoxLayout>
#include <QLabel>
class MainWindow : public QMainWindow
{
Q_OBJECT
public:
explicit MainWindow(QWidget *parent = 0);
};
#endif // MAINWINDOW_H
主窗口.cpp
#include "MainWindow.h"
MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
{
QScrollArea *scrollArea = new QScrollArea(this);
QFrame *frame = new QFrame();
QVBoxLayout *l = new QVBoxLayout(frame);
int N = 121004;
scrollArea->setWidget(frame);
scrollArea->setWidgetResizable(true);
for (int n = 0; n < N; n++) { l->addWidget(new QLabel(QString::number(n), this)); }
resize(640, 480);
setCentralWidget(scrollArea);
}
最佳答案
我有一些坏消息和一些好消息:
坏消息:您无法直接使用 Qt Widgets 执行此操作。
好消息:有一种方法可以做到这一点,它与列表中的项目数量无关(甚至数十亿),但您需要给自己时间来学习如何这样做。
所以,首先:QScrollArea 不是执行此操作的方法。正确的方法是使用 Model/View/Controller programming paradigm .具有要显示的信息的数据模型必须与 View 完全分离,这样 Qt 就只能担心显示用户试图查看的项目。想一想:如果您有十亿个元素要放入该列表中,这是否意味着用户必须一次看到所有元素?这是否意味着 Qt 必须渲染它们?在您的代码中,这就是您要求 Qt 执行的操作。您对它的缓慢感到惊讶吗?
建议编号1:阅读Qt如何管理Model/View programming ,然后选择正确的查看工具。我建议 QListView对于你所描述的。 QTableView如果您可以将事情放在表格中,会让事情变得更容易。
对列表的控制是通过委托(delegate)来完成的。委托(delegate)是负责在 View 中绘制小部件的类。默认只会做文本与图标。
建议编号2: 忘记为每个元素创建 Qt 小部件。 I just finished answering another guy's question为什么这不起作用,即使在使用委托(delegate)时也是如此。看看Qt Torrent Example查看控件是如何在那里绘制的。
您可以做的是绘制控件,而不是小部件。这是因为您创建的每个小部件都必须进入 Qt 中的主事件循环,这会使您的程序变慢(您已经体验过)。如果您从一循环到一百万只是为了添加数字,那将花费大量时间。您真的希望 Qt 的事件循环遍历所有小部件以处理它们中的每一个吗?
建议编号3: 从简单开始!你似乎有很多事情要做。从模型/ View 开始,然后添加一个将绘制自定义控件的委托(delegate),然后展开它。给自己时间学习这一切。
祝你好运!
关于c++ - 如何在 Qt C++ 应用程序中显示带有大量小部件的可滚动列表作为项目?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45389540/
出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits
对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl
如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此
我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende
我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere