19.10 Boost Asio 同步文件传输
在原生套接字编程中我们介绍了利用文件长度来控制文件传输的方法,本节我们将采用另一种传输方式,我们通过判断字符串是否包含goodbye lyshark关键词来验证文件是否传输结束了,当然了这种传输方式明显没有根据长度传输严谨,但使用这种方式也存在一个有点,那就是无需确定文件长度,因为无需读入文件所以在传输速度上要快一些,尤其是面对大文件时。
服务端代码如下所示,在代码中我们分别封装实现recv_remote_file该函数用于将远程特定目录下的文件拉取到本地目录下,而send_local_file函数则用于将一个本地文件传输到对端主机上,这两个函数都接收三个参数,分别是套接字句柄,本地文件与远程文件的文件路径,在传输时采用了while循环读取发送的实现方式每次传输1024个字节,直到传输结束为止。
#include <iostream>
#include <string>
#include <boost/asio.hpp>
#include <boost/array.hpp>
using namespace boost::asio;
// 将远程特定目录下的文件拉取到本地目录
bool recv_remote_file(ip::tcp::socket *socket, std::string remote_file_path, std::string local_file_path)
{
boost::system::error_code error_code;
// 发送需要下载的文件,告诉客户端我需要下载的文件路径
std::string message = remote_file_path;
bool ref = (*socket).write_some(boost::asio::buffer(message), error_code);
if (ref == false)
return false;
char buffer[1024] = { 0 };
// 打开文件,准备写入,保存远程文件到本地
FILE * fp = fopen(local_file_path.c_str(), "wb");
if (NULL == fp)
return false;
int length = 0;
// 每次传输1024字节,直到传输全部结束
while ((length = (*socket).read_some(boost::asio::buffer(buffer, 1024), error_code)) > 0)
{
// 判断最后一次是否为结束符号
if (strncmp(buffer, "goodbye lyshark",15) == 0)
{
std::cout << "传输结束,再见了 lyshark" << std::endl;
fclose(fp);
return true;
}
if (fwrite(buffer, sizeof(char), length, fp) < length)
{
std::cout << "写入文件失败" << std::endl;
break;
}
std::cout << "接收字节数: " << length << " Bytes" << std::endl;
memset(buffer, 0, 1024);
}
if (error_code)
{
fclose(fp);
return false;
}
fclose(fp);
return true;
}
// 将本地特定文件发送到远程的特定目录下
bool send_local_file(ip::tcp::socket *socket, std::string local_file_path, std::string remote_file_path)
{
boost::system::error_code ec;
char buffer[1024] = { 0 };
// 发送放入目标位置
bool ref = (*socket).write_some(boost::asio::buffer(remote_file_path));
if (ref == false)
return false;
// 打开待发送文件
FILE * fp = fopen(local_file_path.c_str(), "rb");
if (NULL == fp)
return false;
int length = 0;
// 每次读入1024字节 直到全部读取结束
while ((length = fread(buffer, sizeof(char), 1024, fp)) > 0)
{
bool ref = (*socket).write_some(boost::asio::buffer(buffer, 1024));
if (ref != false)
{
std::cout << "发送字节数: " << length << " Bytes" << std::endl;
memset(buffer, 0, 1024);
}
}
// 发送结束符
(*socket).write_some(boost::asio::buffer("goodbye lyshark"));
if (ec)
{
fclose(fp);
return false;
}
fclose(fp);
return true;
}
int main(int argc, char* argv[])
{
io_service io_service;
ip::tcp::acceptor acceptor(io_service, ip::tcp::endpoint(ip::tcp::v4(), 6666));
ip::tcp::socket socket(io_service);
acceptor.accept(socket);
std::cout << "远端IP地址: " << socket.remote_endpoint().address() << std::endl;
std::cout << "本端IP地址: " << socket.local_endpoint().address() << std::endl;
// 将远程目录下 c://lyshark.exe 下载到本地的 d://lyshark.exe
bool recv_ref = recv_remote_file(&socket, "c://lyshark.exe", "d://lyshark.exe");
std::cout << "下载状态: " << recv_ref << std::endl;
std::system("pause");
// 将本地目录中的 d://lyshark.exe 发送到远程 c://test.exe
bool send_ref = send_local_file(&socket, "d://lyshark.exe", "c://test.exe");
std::cout << "上传状态: " << send_ref << std::endl;
std::system("pause");
return 0;
}
客户端代码代码如下所示,分别实现了两个函数,函数upload_file用于将本地文件上传到服务器端,函数download_file则用于接收服务器端发送过来的文件,过程中同样采用while循环,每次传输1024个字节。
#include <iostream>
#include <boost/asio.hpp>
#include <boost/array.hpp>
using namespace boost::asio;
// 将本地特定文件发送到远程的特定目录下
bool upload_file(ip::tcp::socket *socket)
{
boost::system::error_code error_code;
boost::array<char, 4096> buf = { 0 };
char buffer[1024] = { 0 };
// 接收要下载文件路径
size_t len = (*socket).read_some(boost::asio::buffer(buf));
if (len == 0)
return false;
// 打开需要发送的文件
FILE * fp = fopen(buf.data(), "rb");
if (NULL == fp)
return false;
int length = 0;
// 每次读入1024字节,直到全部读取结束
while ((length = fread(buffer, sizeof(char), 1024, fp)) > 0)
{
bool ref = (*socket).write_some(boost::asio::buffer(buffer, 1024));
if (ref != false)
{
std::cout << "发送字节数: " << length << " Bytes" << std::endl;
memset(buffer, 0, 1024);
}
}
// 发送结束符
(*socket).write_some(boost::asio::buffer("goodbye lyshark"));
// 如果出现错误直接退出
if (error_code)
{
fclose(fp);
return false;
}
fclose(fp);
return true;
}
// 获取远程发送过来的文件
bool download_file(ip::tcp::socket *socket)
{
boost::system::error_code error_code;
// 读入需要保存文件路径
char file_path[4096] = { 0 };
(*socket).read_some(boost::asio::buffer(file_path));
char buffer[1024] = { 0 };
//打开文件,准备写入
FILE * fp = fopen(file_path, "wb");
if (NULL == fp)
return false;
int length = 0;
// 每次传输1024字节
while ((length = (*socket).read_some(boost::asio::buffer(buffer, 1024), error_code)) > 0)
{
// 判断最后一次是否为结束符号
if (strncmp(buffer, "goodbye lyshark",15) == 0)
{
std::cout << "传输结束,再见了 lyshark" << std::endl;
fclose(fp);
return true;
}
if (fwrite(buffer, sizeof(char), length, fp) < length)
{
printf("写入文件失败 ! \n");
break;
}
printf("接收字节: %d byte \n", length);
memset(buffer, 0, 1024);
}
if (error_code)
{
fclose(fp);
return true;
}
fclose(fp);
return true;
}
int main(int argc, char* argv[])
{
io_service io_service;
ip::tcp::socket socket(io_service);
ip::tcp::endpoint ep(ip::address_v4::from_string("127.0.0.1"), 6666);
socket.connect(ep);
// 该函数对应服务端的 recv_remote_file
bool upload_ref = upload_file(&socket);
std::cout << "上传状态: " << upload_ref << std::endl;
std::system("pause");
// 该函数对应服务端的 send_local_file
bool recv_ref = download_file(&socket);
std::cout << "下载状态: " << recv_ref << std::endl;
std::system("pause");
return 0;
}
读者可自行编译并运行上述程序,代码中依次实现了上传与下载功能,如下图所示;

19.10 Boost Asio 同步文件传输的更多相关文章
- qt+boost::asio+tcp文件传输
客户端: void qt_boost::pbSendFileClicked(){ QString filename = ui.leFileName->text(); QByteArray ba ...
- boost::asio 同步&异步例子
同步客户端: using boost::asio; io_service service; ip::tcp::endpoint ep( ip::address::from_string(); ip:: ...
- Linux文件传输FTP详解
ftp命令用来设置文件系统相关功能.ftp服务器在网上较为常见,Linux ftp命令的功能是用命令的方式来控制在本地机和远程机之间传送文件,这里详细介绍Linux ftp命令的一些经常使用的命令,相 ...
- FTP文件传输协议两种模式 ftp协议集,错误码集,ftp客户端命令集
TCP/IP协议中,FTP标准命令TCP端口号为21,Port方式数据端口为20.FTP协议的任务是从一台计算机将文件传送到另一台计算机,它与这两台计算机所处的位置.联接的方式.甚至是是否使用相同的操 ...
- Windows、Linux -- 远程登录、文件传输、文件共享
Linux <---> Linux 远程登录: ssh服务 ssh root@10.20.62.124 文件传输: scp服务 上传 scp /home/xxx.txt root@1 ...
- 详解:基于WEB API实现批量文件由一个服务器同步快速传输到其它多个服务器功能
文件同步传输工具比较多,传输的方式也比较多,比如:FTP.共享.HTTP等,我这里要讲的就是基于HTTP协议的WEB API实现批量文件由一个服务器同步快速传输到其它多个服务器这样的一个工具(简称:一 ...
- boost::ASIO的同步方式和异步方式
http://blog.csdn.net/zhuky/article/details/5364574 http://blog.csdn.net/zhuky/article/details/536468 ...
- 使用Boost asio实现同步的TCP/IP通信
可以先了解一下Boost asio基本概念,以下是Boost asio实现的同步TCP/IP通信: 服务器程序部分,如果想保留套接字之后继续通信,可以动态申请socket_type,保存指针,因为so ...
- boost::asio设置同步连接超时
boost::asio设置同步连接超时 CSDN上求助无果,只好用自创的非主流方法了.asio自带的例子里是用deadline_timer的async_wait方法来实现超时的,这种方法需要单独写 ...
- 【Linux探索之旅】第四部分第三课:文件传输,潇洒同步
内容简单介绍 .第四部分第三课:文件传输.潇洒同步 2.第四部分第四课:分析网络.隔离防火 文件传输.潇洒同步 这一课的内容相对简单,所以我们慢慢享用. 经过上一课的学习.我们已经知道怎样远程连接到其 ...
随机推荐
- 当谈论 React hook,我们究竟说的是什么?
这个标题很大,但是落点很小,只是我,一个开发者在学习和使用 hooks 中的一点感受和总结. React hook 的由来 React hook 的由来,其实也可以看作是前端技术不断演进的结果. 在 ...
- COGI控制删除按钮增强
1.业务需求 COGI在SAP中主要用于解决生产订单异常的问题.通常发生在生产订单执行过程中,当出现了特定的异常情况时,需要进行处理.这些异常情况可能包括物料缺失.数量不匹配.质量问题等等. 当这些异 ...
- 将应用程序注册为Linux系统服务,并设置开机自启动
1.新建服务文件 vim /usr/lib/systemd/system/statagent.service 其中statagent是自定义的服务名称 2.文件内容 [Unit] Descriptio ...
- C#9.0:Improved Pattern Matching
增强的模式匹配 C#9.0添加了几种新的模式.我们可以参阅模式匹配教程 ,来了解下面代码的上下文: 1 public static decimal CalculateToll(object vehic ...
- JSP | 常见 JSP 简答题
一.简述 JSP 的工作原理 当我们访问一个JSP页面的时候,这个文件首先会被JSP引擎翻译为一个Java源文件,其实就是一个Servlet,并进行编译,然后像其他Servlet一样,由Servlet ...
- 第六届蓝桥杯C++C组 A~F题题解
蓝桥杯历年国赛真题汇总:Here 1. 分机号 X老板脾气古怪,他们公司的电话分机号都是3位数,老板规定,所有号码必须是降序排列,且不能有重复的数位.比如: 751,520,321 都满足要求,而, ...
- L3-020 至多删三个字符 (30分) (DP)
问题描述: 给定一个全部由小写英文字母组成的字符串,允许你至多删掉其中 3 个字符,结果可能有多少种不同的字符串? 输入格式: 输入在一行中给出全部由小写英文字母组成的.长度在区间 [4, 1e6] ...
- 一篇文章教你从入门到精通 Google 指纹验证功能
本文首发于 vivo互联网技术 微信公众号 链接:https://mp.weixin.qq.com/s/EHomjBy4Tvm8u962J6ZgsA作者:Sun Daxiang Google 从 An ...
- java基础(15)--多态
一.多态的含义 1.多种形态.多种状态,指的是编译与运行有不同的状态 2.编译时->静态绑定 3.执行时->动态绑定 4.多类典型场景:父类的引用指向了子类型的对象 二.向下转型与向上 ...
- VirtualBox 设置开机自动在后台启动虚拟机
打开 C:\ProgramData\Microsoft\Windows\Start Menu\Programs\StartUp 新建文件 virtualbox.bat 编写脚本 "C:\Pr ...