Qt在运行时会开启一个主线程,如果没有开启工作线程的话,所有界面上的操作都是在主线程,包括更新界面或者处理数据等操作。我们都知道如果处理数据比较多的话,最好是在单独开启一个线程来处理数据,这样就不会影响主线程的运行。
Windows系统:Windows10
Qt版本:Qt5.15或者Qt6
QT中不建议工作线程中更新界面。
workthread.h
#ifndef WORKTHREAD_H
#define WORKTHREAD_H
#include <QThread>
class MainWindow;
class QLabel;
class WorkThread : public QThread
{
Q_OBJECT
public:
WorkThread(QObject *parent);
~WorkThread();
void setObject(MainWindow *obj);
void setLabel(QLabel *l);
void stop();
protected:
void run();
private:
MainWindow *mainObject = nullptr;
QLabel *label = nullptr;
int i = 0;
volatile bool stopped = false;
};
#endif // WORKTHREAD_H
本文福利,莬费领取Qt开发学习资料包、技术视频,内容包括(C++语言基础,Qt编程入门,QT信号与槽机制,QT界面开发-图像绘制,QT网络,QT数据库编程,QT项目实战,QSS,OpenCV,Quick模块,面试题等等)↓↓↓↓↓↓见下面↓↓文章底部点击莬费领取↓↓
workthread.cpp
#include "workthread.h"
#include "mainwindow.h"
#include <QLabel>
#include <QDebug>
WorkThread::WorkThread(QObject *parent):QThread(parent)
{
stopped = false;
}
WorkThread::~WorkThread()
{
}
void WorkThread::setObject(MainWindow *obj)
{
mainObject = obj;
}
void WorkThread::setLabel(QLabel *l)
{
label = l;
}
void WorkThread::run()
{
qDebug() << "线程开始运行";
while(!stopped)
{
msleep(1);
i++;
//mainObject->runInThread();
label->setText(QString("当前计数为:%1").arg(i));
}
//下次启动
stopped = false;
}
//暂停线程
void WorkThread::stop()
{
stopped = true;
qDebug() << "线程已经暂停";
}
mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QMainWindow>
#include "workthread.h"
QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACE
class WorkThread;
class MainWindow : public QMainWindow
{
Q_OBJECT
public:
MainWindow(QWidget *parent = nullptr);
~MainWindow();
void runInThread();
private slots:
void on_pushButton_clicked();
void on_pushButton_2_clicked();
private:
Ui::MainWindow *ui;
WorkThread *thread = nullptr;
int count = 0;
};
#endif // MAINWINDOW_H
mainwindow.cpp
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "workthread.h"
MainWindow::MainWindow(QWidget *parent)
: QMainWindow(parent)
, ui(new Ui::MainWindow)
{
ui->setupUi(this);
thread = new WorkThread(this);
thread->setObject(this);
thread->setLabel(ui->label);
ui->pushButton->setEnabled(true);
ui->pushButton_2->setEnabled(false);
}
MainWindow::~MainWindow()
{
delete ui;
}
void MainWindow::runInThread()
{
count++;
ui->label->setText(QString("当前计数为: %1").arg(count));
}
//启动线程
void MainWindow::on_pushButton_clicked()
{
if (nullptr != thread)
{
thread->start();
}
ui->pushButton->setEnabled(false);
ui->pushButton_2->setEnabled(true);
}
//暂停线程
void MainWindow::on_pushButton_2_clicked()
{
if (nullptr != thread)
{
thread->stop();
}
ui->pushButton->setDisabled(false);
ui->pushButton_2->setDisabled(true);
}
主界面
workthread.h
#ifndef WORKTHREAD_H
#define WORKTHREAD_H
#include <QThread>
class MainWindow;
class QLabel;
struct Student
{
int id;
char sex;
QString name;
};
//声明元类型
Q_DECLARE_METATYPE(Student)
class WorkThread : public QThread
{
Q_OBJECT
public:
WorkThread(QObject *parent);
~WorkThread();
void stop();
protected:
void run();
signals:
void sigCount(Student s);
private:
int i = 0;
volatile bool stopped = false;
};
#endif // WORKTHREAD_H
workthread.cpp
#include "workthread.h"
#include "mainwindow.h"
#include <QLabel>
#include <QDebug>
WorkThread::WorkThread(QObject *parent):QThread(parent)
{
//注册自定义类型
qRegisterMetaType<Student>();
stopped = false;
}
WorkThread::~WorkThread()
{
}
void WorkThread::run()
{
qDebug() << "线程开始运行";
while(!stopped)
{
i++;
//发送信号
Student s;
s.id = i;
s.sex = 'M';
s.name = "lily";
emit sigCount(s);
msleep(1);
}
//下次启动
stopped = false;
}
//暂停线程
void WorkThread::stop()
{
stopped = true;
qDebug() << "线程已经暂停";
}
mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QMainWindow>
#include "workthread.h"
QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACE
class WorkThread;
class MainWindow : public QMainWindow
{
Q_OBJECT
public:
MainWindow(QWidget *parent = nullptr);
~MainWindow();
private slots:
void on_pushButton_clicked();
void on_pushButton_2_clicked();
void slotSetValue(Student s);
private:
Ui::MainWindow *ui;
WorkThread *thread = nullptr;
};
#endif // MAINWINDOW_H
mainwindow.cpp
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "workthread.h"
MainWindow::MainWindow(QWidget *parent)
: QMainWindow(parent)
, ui(new Ui::MainWindow)
{
ui->setupUi(this);
thread = new WorkThread(this);
ui->pushButton->setEnabled(true);
ui->pushButton_2->setEnabled(false);
connect(thread, &WorkThread::sigCount, this, &MainWindow::slotSetValue);
}
MainWindow::~MainWindow()
{
delete ui;
}
//启动线程
void MainWindow::on_pushButton_clicked()
{
if (nullptr != thread)
{
thread->start();
}
ui->pushButton->setEnabled(false);
ui->pushButton_2->setEnabled(true);
}
//暂停线程
void MainWindow::on_pushButton_2_clicked()
{
if (nullptr != thread)
{
thread->stop();
}
ui->pushButton->setDisabled(false);
ui->pushButton_2->setDisabled(true);
}
//槽函数
void MainWindow::slotSetValue(Student( s))
{
ui->label->setText(QString("当前值为: %1 %2 %3").arg(s.id).arg(s.sex).arg(s.name));
}
主界面
如果要在Qt信号槽中使用自定义类型,需要注意使用qRegisterMetaType对自定义类型进行注册,当然在不跨线程时使用自定义类型signal/slot来传递,可能不会出现什么问题;一旦涉及跨线程就很容易出错,回想下信号槽的作用就是用来对象与对象之间通信的,难免会跨线程,建议在使用自定义类型利用信号槽通信时,最好先通过qRegisterMetaType()将自定义类型进行注册,以免出错。
总结qRegisterMetaType使用方法如下:
1、注册位置:在第一次使用此类链接跨线程的signal/slot之前,一般在当前类的构造函数中进行注册;
2、注册方法:在当前类的顶部包含:#include ,构造函数中加入代码:qRegisterMetaType(“Myclass”);
3、Myclass的引用类型需单独注册:qRegisterMetaType(“Myclass&”);
本文福利,莬费领取Qt开发学习资料包、技术视频,内容包括(C++语言基础,Qt编程入门,QT信号与槽机制,QT界面开发-图像绘制,QT网络,QT数据库编程,QT项目实战,QSS,OpenCV,Quick模块,面试题等等)↓↓↓↓↓↓见下面↓↓文章底部点击莬费领取↓↓
我正在使用i18n从头开始构建一个多语言网络应用程序,虽然我自己可以处理一大堆yml文件,但我说的语言(非常)有限,最终我想寻求外部帮助帮助。我想知道这里是否有人在使用UI插件/gem(与django上的django-rosetta不同)来处理多个翻译器,其中一些翻译器不愿意或无法处理存储库中的100多个文件,处理语言数据。谢谢&问候,安德拉斯(如果您已经在rubyonrails-talk上遇到了这个问题,我们深表歉意) 最佳答案 有一个rails3branchofthetolkgem在github上。您可以通过在Gemfi
我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0
给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack
我想用ruby编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序
我构建了两个需要相互通信和发送文件的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
刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr
我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数
我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R