乐趣区

关于c++:QFtp源码学习及目录下载

背景

须要在 QT5 中进行 FTP 文件下载,并须要反对整目录下载,通过比照抉择,最初决定应用 Qt4 中的 QFtp 来实现咱们的需要。因而决定学习源码,看清构造,做到能真正解决所要面对的问题。

合成源码

Qftp 一共只有四个文件,次要文件是 qftp.cpp,这个文件中,有太多的类,首先按类合成到各自文件中,这样利用官网的示例代码,跑起来后,能够不便的查看代码。

类阐明

  • class QFtpCommand : 此类是对 FTP 命令的封装,将命令与 QIODevice 设施关联起来,并返回一个惟一的标识 ID。
  • class QFtpPI : 此类是对 FTP 协定的封装,processReply 是次要函数,应答服务端响应。
  • class QFtpDTP : 此类是数据操作封装,数据读取、解析、存储都是在此类中解决。
  • class QFtpPrivate : 此类是 QFtp 的实际操作类,被组合到 QFtp 类中,是逻辑解决核心。
  • class QFtp : 此类是外壳,用户间接面对。
  • class QUrlInfo : 此为信息类,存储接管到的每一条文件数据信息。

运行流程

所有的客户端命令被压入到命令堆栈。一个命令运行有两个入口:一是命令被压入堆栈时,若堆栈中只有一条命令,即被运行;二是作响应服务端响应时,类型为 idle 或 not waiting。
每个命运被结构时,都会返回惟一 ID,这是很重要的一点,因为命令大多关联着一本地 IO 设施,在清理 IO 时,要留神与命令对应,因为所有的操作都是异步的。

革新 list 响应

QFtp 列当前目录的原有逻辑是取一条数据就发送一条文件或目录的音讯,这样在咱们间断遍历目录时,无奈分分明是哪个目录下的数据,无奈进行正确的递归。
QFtpDTP::socketReadyRead() – 批改读取目录列表时的信号发送形式

if (pi->currentCommand().startsWith(QLatin1String("LIST"))) {
        QVector<QUrlInfo> infos;                        // 减少 vector 来存储整个目录信息
        while (socket->canReadLine()) {
            QUrlInfo i;
            QByteArray line = socket->readLine();
            if (parseDir(line, QLatin1String(""), &i)) {infos.push_back(i);
                //emit listInfo(i);                     // 原来在循环内,读一条数据发送一个 listInfo 信号
            }
            else {if (line.endsWith("No such file or directory\r\n"))
                    err = QString::fromLatin1(line);
            }
        }
        emit listInfos(infos);                          // 改为在循环外发送新增的 listInfos 信号
    }

FtpWindow::addToList(const QVector<QUrlInfo>& urlInfos) – listInfos 响应批改

    for (int i = 0; i < urlInfos.size(); i++)
    {
        QTreeWidgetItem* item = new QTreeWidgetItem;
        QUrlInfo urlInfo = urlInfos[i];
        if (urlInfo.name().compare(".") != 0) {item->setText(0, urlInfo.name().toLatin1());
            item->setText(1, QString::number(urlInfo.size()));
            item->setText(2, QString::number(urlInfo.isDir()));
            item->setText(3, urlInfo.owner());
            item->setText(4, urlInfo.group());
            item->setText(5, urlInfo.lastModified().toString("MMM dd yyyy"));
            QPixmap pixmap(urlInfo.isDir() ? ":/images/dir.png" : ":/images/file.png");
            item->setIcon(0, pixmap);
            isDirectory[urlInfo.name()] = urlInfo.isDir();
            fileList->addTopLevelItem(item);
        }
    }

目录下载

将 FtpWindow::downloadFile() slot 分解成两个函数,减少 downAllFile(QString rootDir) 来实现目录递归。

void FtpWindow::downloadFile()
{files.clear();      // 初始化本地设施
    downDirs.clear();   // 清空须要下载的目录堆栈

    downAllFile(currentPath);   // 下载具体操作,另一个入口在 list 的响应中
    showProgressDialog();       // 进度条显示}

下载的实在操作函数

void FtpWindow::downAllFile(QString rootDir) {QString thisRoot(rootDir + "/");        // 要下载的父目录
    QList<QTreeWidgetItem*> selectedItemList = fileList->selectedItems();
    for (int i = 0; i < selectedItemList.size(); i++)
    {QString fileName = selectedItemList[i]->text(0);
        if (isDirectory.value(fileName)) {      // 若是子目录,组合实现的目录,压入待下载目录堆栈
            if(fileName != "..")
                downDirs.push(thisRoot + fileName);
        }
        else {downloadTotalBytes += selectedItemList[i]->text(1).toLongLong();    // 统计须要下载的字节量
            ...
            QFile* file = new QFile(dirTmp.append("/").append(fileName));
            // 文件下载申请,是异步操作
            int id = ftp->get(QString::fromLatin1((selectedItemList[i]->text(0)).toStdString().c_str()), file);
            files.insert(id, file);     // 本地 IO 设施与其命令绑定并存储
        }
    }
    if (downDirs.size() > 0) {      // 待下载目录堆栈不空,解决一条
        enterSubDir = true;         // 示意正在下载目录
        QString nextDir(downDirs.pop());  // 取须要解决的下一个目录
        ftp->cd(nextDir);                 // 切换到这个目录
        currentDownPath = nextDir;
        ftp->list();                        // 列目录,在其响应中将再递归调用本函数~~~~}
}

list 响应的递归解决局部

    if (!enterSubDir) {     // 下载的文件中没有目录
        ...     
    }
    else {                              // 正解决于目录下载中
        fileList->selectAll();          // 选中列表中所有
        downAllFile(currentDownPath);   // 递归调用下载处理函数
    }

我的项目地址

https://github.com/zhoutk/qtDemo

命令行编译

git clone https://github.com/zhoutk/qtDemo
cd qtDemo/ftpClient & mkdir build & cd build
cmake ..
cmake --build .      

编译时留神:cmake 默认为 x86 架构,须要与你装置的 Qt 版本对应;编译好了,运行前,请留神目录构造是否正确。

小结

我抉择的这种目录下载方式比拟麻烦,整个运行过程都是异步的,调试也比拟难,其中进度条控件还有些问题,其在 5.12.9 与 5.5.1 中的体现还不一样,当初在 5.5.1 中运行还有些小瑕疵,但正确性能保障。过程艰巨,播种颇多。

退出移动版