zoukankan      html  css  js  c++  java
  • [C++]Qt 如何处理密集型耗时的事情(频繁调用QApplication::processEvents)

    https://www.cnblogs.com/senior-engineer/p/5598133.html

    https://www.cnblogs.com/findumars/p/5607683.html

    api含义:QCoreApplication::processEvents() 将处理所有事件队列中的事件并返回给调用者。

    问题描述:

    当主线程在某个槽函数里正在执行processEvents时, 刚好有一个能响应此槽函数的信号发送过来了(肯定是其他线程发的信号),  这时就可能会发生可怕的递归, 

    导致栈溢出崩溃。 原因是processEvents,进入到无尽的递归中。

    示例代码

    bugThread.h

    #include <QThread>
    class BugThread : public QThread
    {
    	Q_OBJECT
    public:
    	BugThread(QObject* parent) : QThread(parent) {}
    signals:
    	void sigBugsignal();
    public:
    	void run() 
    	{
    		while(true)
    		{
    			emit sigBugsignal();
    		}
    	}
    };

    bugs.h & bugs.cpp

    class Bugs : public QMainWindow
    {
    	Q_OBJECT
    
    public:
    	Bugs(QWidget *parent = 0, Qt::WFlags flags = 0);
    	~Bugs();
    	public slots:
    		void onBugSlot();
    private:
    	Ui::BugsClass ui;
    };
    
    
    Bugs::Bugs(QWidget *parent, Qt::WFlags flags)
    	: QMainWindow(parent, flags)
    {
    	ui.setupUi(this);
    	BugThread* bt = new BugThread(this);
    	connect(bt, SIGNAL(sigBugsignal()), this, SLOT(onBugSlot()));
    	bt->start();
    }
    
    Bugs::~Bugs()
    {
    
    }
    void Bugs::onBugSlot()
    {
    	Sleep(1); 
    	QApplication::processEvents();
    }
    ===========================================================================================

    有时候需要处理一些跟界面无关的但非常耗时的事情,这些事情跟界面在同一个线程中,由于时间太长,导致界面无法响应,处于“假死”状态。例如:在应用程序中保存文件到硬盘上,从开始保存直到文件保存完毕,程序不响应用户的任何操作,窗口也不会重新绘制,从而处于“无法响应”状态,这是一个非常糟糕的体验 。

         在这种情况下,有一种方法是使用多线程,即在子线程中处理文件保存,主线程负责界面相关。

         而如果不想使用多线程,最简单的办法就是在文件保存过程中频繁调用QApplication::processEvents()。该函数的作用是让程序处理那些还没有处理的事件,然后再把使用权返回给调用者。

    代码如下:

    复制代码
    复制代码
    bool MyApp::writeFile(const QString &filename)
    {
         QFile file(filename);
    ...
        QApplication::setOverrideCursor(Qt::WaitCursor);
         for(int r = 0; r != rowCount; ++r)
         {
              for(int c = 0; c != colCount; ++c)
              {
                   out << table(r,c);   
                   qApp.processEvents();
              }
         }
        QApplication::restoreOverrideCursor();
    }
    复制代码
    复制代码

         这样一来,程序就能响应了。

         但是,该方法有一个问题:可能正在保存文件的过程中,用户不小心又单击了保存,或不小心关闭了程序主窗口,这样会产生意想不到的后果。

         解决这个问题的最简单的办法是替换成:

    qApp->processEvents(QEventLoop::ExcludeUserInputEvents);//它可以忽略用户的输入(鼠标和键盘事件)。

         进一步的,如果想显示一个带有进度条的对话框,随时显示当前的进度状态,可以使用QProgressDialog。

    复制代码
    复制代码
    bool MyApp::writeFile(const QString &filename)
    {
         QFile file(filename);
    ...
        QApplication::setOverrideCursor(Qt::WaitCursor);
         QProgressDialog progress;
         progress.setWindowTitle(tableData->sNameCH);
         progress.setLabelText(QStringLiteral("数据保存中,请稍候..."));
         //progress.setCancelButton(0);//不显示“取消”按钮
         progress.setCancelButtonText("取消");
         progress.setRange(0,rowCount );
         progress.setModal(true);
         //此处没有调用show()来显示,是因为QProgressDialog会自动决定是否显示
         //如果时间过短,就不会显示。
         for(int r = 0; r != rowCount; ++r)
         {
              progress.setValue(row);
              //如果用户单击了“取消”,就取消保存文件,并删除该文件。
              if(progress.wasCanceled)
              {
                   file.remov();
                   return false;
              }
              for(int c = 0; c != colCount; ++c)
              {
                   out << table(r,c);   
                   qApp.processEvents();
              }
         }
        QApplication::restoreOverrideCursor();
    }
  • 相关阅读:
    IIS 下配置无后缀的URL ReWrite
    获得代理IP或客户端Ip
    asp.net发送邮箱
    URLRewrite伪静态与AspNetPager分页控件的结合
    邮箱大全
    注册表单验证正则表达式
    aspx中伪静态的实现
    JMail发送邮件
    IE 6中负的margin值导致出界部分不显示问题的解决
    Culminis为每个用户组提供的免费TechNet Plus Direct订阅
  • 原文地址:https://www.cnblogs.com/FKdelphi/p/10756006.html
Copyright © 2011-2022 走看看