Qt中实现QThread线程同步QFtp

移动开发
本文讲述的是在Qt中如何实现QThread线程同步QFtp,那么对于QFtp,它是一个用来实现FTP协议的类,#include <qftp.h> 继承了QNetworkProtocol。本文由介绍。

Qt中实现QThread线程同步QFtp ,对于QFtp,它是一个用来实现FTP协议的类,详情查阅资料。接触Qt没有多长时间,但简单几个小例子已经让我感受到Qt在C++运用方面的强大。写了一个小程序,需要在一个单独的线程中使用QFtp来获取FTP服务器上面的文件。FTP是FileZilla。

有两个问题我比较关心:

(1)QThread到底如何使用

(2)QFtp是Async(异步)操作,也就是说例如connectToHost这样的函数都是立刻返回,当操作完成后QFtp会发出signal。然而既然我的Ftp操作是在一个单独的线程,我想写一个函数downloadFtpFile() 来完成从connect到login到下载文件等一系列的操作,然后再返回。相当于我需要Sync(同步)的操作,所以需要等待(block)每个Ftp命令的结果。

在该文章最后有一个推荐的使用QThread的方法。我在这里想补充一点:obj.moveToThread(&thread); 这句话将obj从主线程移动到了thread对象所在的线程。但如果obj的对象里面有其他的变量,那么这些变量是在主线程中生成的。所以如果这些变量中有类变量,不能将obj的this指针作为parent传给他们。

对于第二个问题,我使用了QSemaphore类来完成我的block和同步操作:在slot函数里面接收QFtp命令执行结果的signal,释放信号,同时downloadFtpFile()函数里在调用完每一个QFtp异步命令后等待信号。在有点令人失望的是QSemaphore在通过tryAcquire()等待信号的时候是不处理事件event的。但是我需要在等待的时候程序也能触发slot,告诉我当前命令的执行情况。所以我使用了一个小循环,里面调用qApp->processEvents();来让我的slot函数被触发。下面是代码例子(只是样例,并不完全符合C++语法):

首先是我的下载Ftp文件的函数:

 

  1. downloadFtpFile () //该函数在单独线程里执行     
  2. {     
  3.      int m_idFtpOp; // 该变量用来存放每一个QFtp命令ID     
  4.      int nVal;     
  5.      QFtp*pFtp=newQFtp (this); // 生成QFtp工具对象     
  6.      connect (pFtp,SIGNAL(listInfo(QUrlInfo)),this,SLOT(slotFtpListInfo(QUrlInfo))); // 我们需要listinfo,因为我们需要下载ftp所有当前目录文件     
  7.      connect (pFtp,SIGNAL(commandFinished(int,bool)),this,SLOT(slotFtpCmdFinished(int,bool)));   
  8. // 每个QFtp命令完成之后,会发出commandFinished信号,我们在槽函数中处理该信号  
  9.      m_idFtpOp = pFtp->connectToHost (<FTP地址>, 21); // 连接到远程FTP Server     
  10.      bRet=false;     
  11.      nVal=100;     
  12.      while (bRet == false) // 使用nVal变量来做一个10000ms(10s)的超时     
  13.      {     
  14.            nVal--;     
  15.            if (nVal == 0)     
  16.                 break;     
  17.            qApp->processEvents();   // 这里每100ms处理一次event,使slot函数能够被调用     
  18.            bRet=m_SemOp.tryAcquire (1,100); // 等待信号100ms     
  19.      }     
  20.     if (!bRet || m_bFtpOpError)   // 如果超时,或者slot函数中将m_bFtpOpError置成true,则关闭Ftp,返回错误     
  21.       {     
  22.            pFtp->abort();     
  23.            pFtp->deleteLater();     
  24.            return ERRCODE_FCC_FTP_CONN_TIMEOUT;     
  25.      }     
  26. }   

下面是槽函数

  1. slotFtpCmdFinished (int id, bool error)     
  2. {     
  3.          if (m_idFtpOp == id)  // 如果返回的id是当前正在操作的命令     
  4.          {     
  5.                   if (error)     
  6.                            m_bFtpOpError=true;     
  7.                   else    
  8.                            m_bFtpOpError=false;     
  9.                   m_SemOp.release();  // 释放信号(使downloadFtpFile函数中m_SemOp.tryAcquire()返回true)     
  10.          }     
  11. }   

以上的代码只演示了对QFtp第一个命令connectToHost的等待过程。下面的login,list,get等操作都使用这个方法。

注意:在此例中,QFtp是在当前线程生成的,所以信号listInfo(QUrlInfo)的connect方式是direct连接。如果QFtp是在另一个线程生成(比如说是在函数downloadFtpFile所在类的构造函数中),那么第一:不能将this指针作为parent传给QFtp对象,第二:需要使用qRegisterMetaType<QUrlInfo>("QUrlInfo");来注册QUrlInfo类,因为信号发射与接收在不通的线程中,信号使用queued的方式。如果不注册QURlInfo类,会在运行时动态报告错误。

总结:本文介绍的是在Qt中如何实现QThread线程同步QFtp ,看过本文之后,如果对于QThread不了解的话,那么请参考Qt中QThread使用方法这篇文章。使用本文介绍的方法,可以在独立的线程中用同步的方式使用QFtp。在某些场合,尤其是采用应答机制的系统中,这样的实现可以很大程度上简化程序流程。

【编辑推荐】

浅谈Qt中多线程编程

Qt环境变量配置与安装

在Linux中设置Qt环境变量

用C#实现HTTP协议下的多线程文件传输

C#常用线程同步方法应用场景和实现原理

责任编辑:zhaolei 来源: 互联网
相关推荐

2011-06-14 10:18:58

QThread Qt 线程

2011-06-30 18:15:36

Qt 线程 同步

2011-06-20 13:43:08

Qt Socket 线程

2011-06-24 11:03:31

Qt 多线程 线程

2011-06-24 11:12:39

Qt 多线程 线程

2011-06-30 17:51:17

QT 元类型 线程

2011-06-22 15:42:18

QT 信号

2011-06-22 15:24:50

Qt 线程

2011-06-13 10:03:19

Qt 多线程 编程

2011-06-30 11:23:29

Qt 线程

2011-06-30 17:31:32

Qt 多线程 信号

2010-03-18 14:36:46

Java线程同步

2011-06-13 17:46:07

Qt 串口通信

2010-01-21 11:27:30

linux多线程机制线程同步

2011-06-22 10:12:08

Qt 线程

2010-03-03 17:52:31

Python线程同步

2009-07-01 17:34:03

Servlet和JSP

2009-03-24 08:56:23

数据同步多线程Java

2010-04-21 15:20:31

Unix线程

2011-06-28 15:18:45

Qt 单例模式
点赞
收藏

51CTO技术栈公众号