c++ QNetworkAccessManager及poco 两种方式上传文件到ftp Server

目录

一  各种方式比较 

二 QNetworkAccessManager

三 POCO实现方式

3.1 POCO库下载与编译

3.2 本地ftp server下载

3.3 ftp文件上传

 四 总结


c++ ftp上传文件有三种方式,分别有

  1. Qt4 QFtp

  2. Qt5 QNetworkAccessManager

  3. POCO FTPClientSession

一  各种方式比较 

其中QFtp在QT5中已经丢弃了,无法使用,且源码编译,需要解决编码问题,异常处理不友好; QNetworkAccessManager对于一些文件操作不能实现,如创建目录等;

POCO库功能齐全,可以进行文件操作、上传方式改变等。

下面列出QNetworkAccessManager及poco两种实现方式

二 QNetworkAccessManager

这里仅用到文件上传的功能,所以代码主要功能包含url设置和文件上传ftpPut;

FtpQNetWork.h 

class FtpQNetWork : public FtpBase {
 public:
  FtpQNetWork();
  virtual ~FtpQNetWork();
  virtual bool ftpPut(const QString &localPath, const QString &serverPath,
                      const QJsonObject &obj = QJsonObject());

 private:
  void setFtpInfos(const QJsonObject &obj, QUrl &url);

 private:
  QNetworkAccessManager mNetworkMan;
};

FtpQNetWork.cpp

#include "FtpQNetWork.h"

#include <QDebug>
#include <QEventLoop>
#include <QFile>
#include <QNetworkReply>
#include <QUrl>

#include "common/common.h"

FtpQNetWork::FtpQNetWork() {}

FtpQNetWork::~FtpQNetWork() {}

bool FtpQNetWork::ftpPut(const QString &localPath, const QString &serverPath,
                         const QJsonObject &obj) {
  QFile file(localPath);
  if (!file.open(QFile::ReadOnly)) {
    qDebug() << "file open failed=" << localPath;
    return false;
  }
  QUrl url;
  url.setScheme("ftp");  //设置通讯协议
  if (obj.isEmpty()) {
    setFtpInfos(mJsonFtpCfg, url); 
  } else {
    setFtpInfos(obj, url);
  }
  url.setPath(serverPath);
  qDebug() << "url " << url;
  QNetworkReply *reply = mNetworkMan.put(QNetworkRequest(url), file.readAll());
  if (reply) {
   //通过QEventLoop 阻塞主线程,直到文件上传完成
    QEventLoop loop; //将上传文件操作置为同步操作,等待ftp操作完成再执行其他任务
    QObject::connect(reply, &QNetworkReply::finished, &loop, &QEventLoop::quit);
    loop.exec();
    if (reply->error() != QNetworkReply::NoError) {
      return false;
    } else {
      int code =
          reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
      if (code != 200) {
        return false;
      }
    }
    reply->deleteLater();

  } else {
    qDebug() << "ftpPut reply nullptr donothing";
  }
  return true;
}
//设置用户名密码端口号等信息
void FtpQNetWork::setFtpInfos(const QJsonObject &obj, QUrl &url) {
  if (pCom->checkKeyExist(obj, "ip/port/userName/pwd")) {
    auto &&ip = obj.value("ip").toString();
    auto &&port = obj.value("port").toInt();
    auto &&pwd = obj.value("pwd").toString();
    auto &&userName = obj.value("userName").toString();
    url.setHost(ip);
    url.setPort(port);
    url.setPassword(pwd);
    url.setUserName(userName);
  } else {
    qDebug() << "not contain ip/port/userName/pwd=" << obj;
  }
}

三 POCO实现方式

poco的方式实现需要下载对应的库 编译后使用

3.1 POCO库下载与编译

我是按照这个做的,仅供参考Windows下Poco库的编译和安装_poco 编译-CSDN博客

这个是用系统编译的库,如果使用QT编程,下载源码后尽量在QT中进行编译,以防有些dll不能使用 。我就是系统编译后在项目中使用编译不通过,然后再QT中又编译一遍才可以的。编译完成后,将bin,include,lib文件夹放入项目中,便可以在code中直接引用该库,如下我放的位置:

3.2 本地ftp server下载

http://learning.happymmall.com/ftpserver/FTPServer.rar

下载安装后,便可使用,很简洁

设置好用户密码、共享目录及端口后,便可以使用测试。

注意:即使上面server中设置了共享目录,但在服务器中其实看不到D:\software\ftpTest这一层目录,只能看到最终上传的文件本身,也就是说上传的是服务器根目录,如下图;当然在本地的D:\software\ftpTest是有该层目录的,可以说二者的文件是共通的。

服务器:

本地目录:

3.3 ftp文件上传

FtpPoco.h

namespace Poco {
namespace Net {
class FTPClientSession;
}
}  // namespace Poco

class FtpPoco : public FtpBase {
 public:
  FtpPoco();
  virtual ~FtpPoco();
  virtual bool login(const QJsonObject &para);
  virtual bool ftpPut(const QString &localPath, const QString &serverPath,
                      const QJsonObject &obj = QJsonObject());

 private:
  std::shared_ptr<Poco::Net::FTPClientSession> mClient;
};
#endif  // FTPPOCO_H

FtpPoco.cpp

之前的代码对上传结果判断是无效的,因为那只能检测到本地文件是否存在,并不能判断远端ftp服务器中某个文件是否存在,修改后使用try catch的方式进行结果反馈;

#include "FtpPoco.h"

#include <Poco/Net/FTPClientSession.h>
#include <Poco/StreamCopier.h>

#include <QString>
#include <fstream>
#include <iostream>

#include "common/common.h"

FtpPoco::FtpPoco() {
  mClient = std::make_shared<Poco::Net::FTPClientSession>();
}
FtpPoco::~FtpPoco() {}

bool FtpPoco::login(const QJsonObject &para) {
  if (mClient) {
    if (pCom->checkKeyExist(para, "ip/port/userName/pwd")) {
      auto &&port = para.value("port").toInt();
      auto &&pwd = para.value("pwd").toString().toStdString();
      auto &&ip = para.value("ip").toString().toStdString();
      auto &&userName = para.value("userName").toString().toStdString();
      //使用try catch的方式辅助判断登录状态
      try {
        if (!mIsLoginFlag) {
       //使用 用户名 密码 端口 ip 建立ftp连接
          mClient->open(ip, port, userName, pwd);
        }
       //登录
        mIsLoginFlag = mClient->isLoggedIn();
        if (mIsLoginFlag) {
          qDebug() << "login success";
        } else {
          qDebug() << "login failed";
        }
      } catch (std::exception &e) {
        qDebug() << "login exception e=" << e.what();
        mIsLoginFlag = false;
      }
    }
  } else {
    qDebug() << "mClient nullptr";
  }
  return mIsLoginFlag;
}

bool FtpPoco::ftpPut(const QString &localPath, const QString &serverPath,
                     const QJsonObject &obj) {
  (void)obj;
  login(mJsonFtpCfg);
  bool flag = false;
  auto &&srcFn = localPath.toStdString();
  auto &&destFn = serverPath.toStdString();
//poco中上传文件 没有可以判断结果状态的方法,这里利用try catch来进行判断上传结果
  try {
    auto &&newDir = destFn.substr(0, destFn.find_last_of("/"));
    if (mClient) {
      mClient->setWorkingDirectory(newDir);
      std::ifstream ifStr(srcFn, std::ios::binary);
      std::ostream &ostr = mClient->beginUpload(destFn);
      ostr << ifStr.rdbuf();
      mClient->endUpload();
      flag = true;
      ifStr.close();
    }
  } catch (std::exception &e) {
    qDebug() << "has a exception=" << e.what();
  }
  return flag;
}

 四 总结

代码中可以观察到二者共有一个父类FtpBase,将两种方式封装起来,便于后续的使用。

FtpBase.h

#ifndef FTPBASE_H
#define FTPBASE_H

class QString;
#include <QJsonObject>
#include <QObject>
class FtpBase : public QObject {
 public:
  FtpBase();
  virtual ~FtpBase();
  virtual void init(const QJsonObject &ftpObj);
  virtual bool login(const QJsonObject &para);
  virtual bool ftpPut(const QString &localPath, const QString &serverPath,
                      const QJsonObject &obj = QJsonObject());

 protected:
  bool mIsLoginFlag;
  QJsonObject mJsonFtpCfg;
};
#endif  // FTPBASE_H

FtpBase.cpp

#include "FtpBase.h"

#include <QString>

#include "common/common.h"

FtpBase::FtpBase() : mIsLoginFlag(false) {}
FtpBase::~FtpBase() {}
//调用init方法 用于登录信息的初始化
void FtpBase::init(const QJsonObject &ftpObj) {
  mJsonFtpCfg = ftpObj;
  qDebug() << "mJsonFtpCfg=" << mJsonFtpCfg;
}

bool FtpBase::login(const QJsonObject &para) {
  (void)para;
  return false;
}

bool FtpBase::ftpPut(const QString &localPath, const QString &serverPath,
                     const QJsonObject &obj) {
  (void)localPath;
  (void)serverPath;
  (void)obj;
  return false;
}

 注意:若使用CMake编译配置,设置Poco的使用库时要注意当前编译模式Debug or release等,不同编译模式压迫使用对应的库

set(pocoNetPre ${thirdpartyPre}/Poco/${CMAKE_BUILD_TYPE})
if(CMAKE_BUILD_TYPE STREQUAL Debug)
    set(pocoNetLib ${pocoNetPre}/lib/PocoNetd.lib)
    set(pocoFoundationLib ${pocoNetPre}/lib/PocoFoundationd.lib)
else()
    set(pocoNetLib ${pocoNetPre}/lib/PocoNet.lib)
    set(pocoFoundationLib ${pocoNetPre}/lib/PocoFoundation.lib)

持续更新。。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: QNetworkAccessManager 是一个 Qt 框架中提供的网络访问管理器,它可以用于访问不同的网络资源,包括 HTTP、FTP 等。但是,QNetworkAccessManager 并没有提供重命名或删除 FTP 文件的方法,这些功能需要使用 FTP 协议提供的相关命令实现。 要实现 FTP 文件的重命名或删除操作,可以使用 Qt 提供的 QFtp 类。QFtp 类提供了一组方法,可以通过 FTP 协议实现文件的上传、下载、重命名和删除等操作。以下是使用 QFtp 类实现 FTP 文件重命名和删除的示例代码: ```cpp #include <QtFtp> QFtp *ftp = new QFtp(this); ftp->connectToHost("ftp.example.com", 21); // 连接 FTP 服务器 ftp->login("username", "password"); // 登录 FTP 服务器 ftp->rename("oldfile.txt", "newfile.txt"); // 重命名文件 ftp->remove("file.txt"); // 删除文件 ``` 在上面的示例代码中,首先创建了一个 QFtp 对象,并使用 connectToHost() 方法连接到 FTP 服务器。然后,使用 login() 方法登录到 FTP 服务器。最后,使用 rename() 方法重命名文件,使用 remove() 方法删除文件。 需要注意的是,QFtp 类已经被标记为弃用,建议使用更现代的 Qt 网络模块 QNetworkAccessManager 来实现网络访问功能。如果需要在 Qt 中进行 FTP 文件的重命名和删除操作,建议使用第三方的 FTP 库,如 libcurl 等。 ### 回答2: QNetworkAccessManager是Qt网络模块中的类,用于处理网络请求和响应。它提供了一些内置的协议,如HTTP、HTTPS和FTP,用于进行相应的网络通信。 对于FTP协议而言,QNetworkAccessManager提供了基本的上传和下载功能,但是无法直接实现FTP的重命名和文件删除操作。 要实现FTP的重命名和文件删除功能,我们需要使用QNetworkAccessManager配合QUrl和QNetworkRequest等类来自行实现。具体的步骤如下: 1. 首先,创建一个QNetworkAccessManager对象,并连接相关的信号和槽,以便处理请求和响应。 2. 定义FTP服务器的地址和端口,并使用QUrl类来构造FTP的目标地址。 3. 通过QNetworkRequest类设置请求的URL、请求方法和其他必要的头部信息。 4. 对于重命名操作,可以使用QNetworkAccessManager的upload()函数来上传一个空的文件,同时设置请求方法为RENAME,并在请求的URL中指定原文件名和新文件名。 5. 对于文件删除操作,可以使用QNetworkAccessManager的deleteResource()函数来发送一个删除请求,并在请求的URL中指定要删除的文件名。 6. 最后,使用QNetworkAccessManager的sendCustomRequest()函数发送请求,然后处理响应结果。 需要注意的是,FTP服务器的支持程度可能因服务器的实现而有所不同,某些服务器可能不支持重命名或文件删除操作。因此,在使用QNetworkAccessManager进行FTP操作时,需要根据实际情况进行相应的检查和处理。 综上所述,QNetworkAccessManager无法直接实现FTP的重命名和文件删除操作,但通过自行构建请求并发送到FTP服务器,我们可以实现这些功能。 ### 回答3: QNetworkAccessManager是Qt框架中用于处理网络请求的类,它提供了一种方便的方式来发送HTTP和HTTPS请求,但不支持直接的FTP操作。因此,使用QNetworkAccessManager无法直接实现FTP的重命名和文件删除功能。 实现FTP的重命名和文件删除功能可以通过其他方式来实现,例如使用QFTP类或者使用第三方的FTP库。 1. 使用QFTP类:QFTP是一个封装了FTP协议的Qt类,它提供了一系列的方法来实现FTP的操作,包括重命名和文件删除。可以使用QFTP类中的rename()方法来实现重命名功能,使用remove()方法实现文件删除功能。 2. 使用第三方的FTP库:Qt框架也支持使用第三方的FTP库来实现FTP操作。可以选择一些开源的FTP库,例如libcurl或libftp等,在项目中引入并使用这些库来实现FTP的重命名和文件删除功能。 使用这些方式,可以在Qt应用程序中实现FTP的重命名和文件删除功能。但需要注意的是,使用第三方库可能需要进行额外的配置和安装,并且代码的实现可能会与Qt自带的类有所不同。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值