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列当前目录的原有逻辑是取一条数据就发送一条文件或目录的消息,这样在我们连续遍历目录时,无法分清楚是哪个目录下的数据,无法进行正确的递归。这样改造效率应该会好一些且完全控制整个目录的显示,比如,目录显示在上面。当然也可以将递归放到commandFinished消息响应中去。

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版本对应;编译好了,运行前,请注意目录结构是否正确。

小结

我选择的这种目录下载方式比较麻烦,没有放到后台再开一个进程去处理,试图做整体考虑,且整个运行过程都是异步的,调试也比较难,其中进度条控件控制还有些坑,需要小心处理。过程艰难,收获颇多。

©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容