19.11 Boost Asio 获取远程目录
远程目录列表的获取也是一种很常用的功能,通常在远程控制软件中都存在此类功能,实现此功能可以通过filesystem.hpp库中的directory_iterator迭代器来做,该迭代器用于遍历目录中的文件和子目录,它允许开发者轻松遍历目录层次结构并对遇到的文件和目录执行各种操作。
使用directory_iterator构造函数创建一个迭代器,该迭代器指向目录中的第一个条目。然后,我们使用一个范围for循环来遍历目录中的每个文件和目录,并使用pos->path().string()方法即可得到该目录下的文件路径。
当然,directory_iterator仅遍历目录的直接子文件夹。如果想遍历目录层次结构中的所有文件和目录,则需要改用recursive_directory_iterator来实现递归遍历,或者自行拼接路径完成遍历。
服务端代码实现如下,在代码中我们通过write_some发送需要获取远程目录字符串,接着通过调用一次read_some接收一次需要循环的次数,当收到客户端传来的目录列表数量后,则可以在循环内依次调用read_some函数读取数据,并格式化为CatalogData类型,并输出其变量中的数据。
#include <iostream>
#include <boost/lexical_cast.hpp>
#include <boost/asio.hpp>
// 定义结构体,将完整数据放入结构体中
typedef struct
{
char file_path[2048];
bool is_directory;
long file_size;
}CatalogData;
using namespace std;
using namespace boost;
using namespace boost::asio;
CatalogData ptr;
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);
boost::system::error_code error_code;
// 发送需要获取的目录
socket.write_some(boost::asio::buffer("d://lyshark"), error_code);
// 接收文件数量
char recv_count[1024] = { 0 };
size_t len = socket.read_some(boost::asio::buffer(recv_count), error_code);
std::cout << "接收到数量: " << recv_count << std::endl;
// 判断如果数量大于等于1则需要循环获取
int for_count = lexical_cast<int>(recv_count);
if (for_count >= 1)
{
// 循环接收目录
for (int x = 0; x < for_count; x++)
{
// 准备接收缓冲区,以及接收数据包
char recv_catalogdata[sizeof(CatalogData)] = { 0 };
socket.read_some(boost::asio::buffer(recv_catalogdata, sizeof(CatalogData)), error_code);
// 将收到的字节序转换为CatalogData结构体
CatalogData *data = (CatalogData *)recv_catalogdata;
std::cout << "文件路径: " << data->file_path
<< "是否为目录: " << data->is_directory
<< "文件大小: " << data->file_size
<< std::endl;
memset(recv_catalogdata, 0, sizeof(CatalogData));
}
}
std::system("pause");
return 0;
}
客户端代码实现如下,相对于服务端客户端的实现则变得复杂一些,在代码中首先通过read_some函数调用获取到服务器端传来的目录C://usr信息,接着直接调用GetFileState函数并以此获取到当前目录下的文件列表信息,并将此信息存储到send_buffer容器内存储,接着再次调用write_some发送此容器内有多少行数据,最后通过使用循环的方式依次write_some发送send_catalogdata文件列表,直到全部发送结束则退出程序。
#include <iostream>
#include <string>
#include <fstream>
#include <vector>
#include <boost/filesystem.hpp>
#include <boost/asio.hpp>
#include <boost/lexical_cast.hpp>
using namespace std;
using namespace boost;
using namespace boost::asio;
using namespace boost::filesystem;
// 定义结构体,将完整数据放入结构体中
typedef struct
{
char file_path[2048];
bool is_directory;
long file_size;
}CatalogData;
// 利用流获取文件大小
long GetFileSize(std::string filename)
{
long ref_kb;
std::ifstream ptr(filename, std::ios::in | std::ios::binary);
if (ptr.is_open() == true)
{
ptr.seekg(0, std::ios::end); // 移动到末尾
ref_kb = ptr.tellg(); // 获取字节数
ptr.close();
return ref_kb;
}
return 0;
}
// 遍历文件函数,并将结果存入RefVect
void GetFilePath(const string& pathName, std::vector <std::string> &RefVect)
{
directory_iterator end;
for (directory_iterator pos(pathName); pos != end; ++pos)
{
RefVect.push_back(pos->path().string());
}
}
// 获取到当前目录详细信息,并依次取出数据
std::vector<CatalogData> GetFileState(const string& pathName)
{
std::vector < std::string > ref_file_path;
GetFilePath(pathName,ref_file_path);
// 循环获取目录属性
std::vector<CatalogData> ref_date;
for (int x = 0; x < ref_file_path.size(); x++)
{
CatalogData ptr;
// 判断如果是目录,则不计算文件大小
if (is_directory(ref_file_path[x]))
{
ptr.is_directory = true;
ptr.file_size = 0;
strcpy(ptr.file_path, ref_file_path[x].c_str());
}
// 如果是文件则计算大小
else
{
ptr.is_directory = false;
ptr.file_size = GetFileSize(ref_file_path[x]);
strcpy(ptr.file_path, ref_file_path[x].c_str());
}
// 依次放入容器
ref_date.push_back(ptr);
}
return ref_date;
}
CatalogData ptr;
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);
boost::system::error_code error_code;
socket.connect(ep, error_code);
// 接收需要获取的目录地址
char recv_buf[1024] = { 0 };
size_t len = socket.read_some(boost::asio::buffer(recv_buf), error_code);
std::cout << "服务端需要获取: " << recv_buf << std::endl;
// 判断目录是否存在,存在则执行遍历目录
if (filesystem::exists(recv_buf))
{
std::vector<CatalogData> send_buffer;
send_buffer = GetFileState(recv_buf);
// 判断如果获取到了目录,则发送目录个数
if (send_buffer.size() > 0)
{
// 发送目录总数
std::string send_count = lexical_cast<std::string>(send_buffer.size());
std::cout << "获取到的目录文件数量: " << send_count << std::endl;
socket.write_some(boost::asio::buffer(send_count), error_code);
}
// 开始循环发送目录
for (int x = 0; x < send_buffer.size(); x ++)
{
// 填充局部结构并发送
char send_catalogdata[sizeof(CatalogData)] = { 0 };
// 拷贝将send_buffer依次赋值到ptr指针上
strcpy(ptr.file_path, send_buffer[x].file_path);
ptr.is_directory = send_buffer[x].is_directory;
ptr.file_size = send_buffer[x].file_size;
// 将字节序拷贝到send_catalogdata中
memcpy(send_catalogdata, &ptr, sizeof(CatalogData));
// 开始发送字节序
socket.write_some(boost::asio::buffer(send_catalogdata), error_code);
memset(send_catalogdata, 0, sizeof(CatalogData));
}
}
std::system("pause");
return 0;
}
至此读者可自行编译上述代码,并以此先运行服务端程序,接着再运行客户端程序,此时即可获取到远程主机中d://lyshark目录下的所有文件信息,文件属性,以及文件大小,输出效果如下图所示;

19.11 Boost Asio 获取远程目录的更多相关文章
- fabric动态获取远程目录列表
#!/usr/bin/pythonfrom fabric.api import *env.user='root'env.hosts=['172.10.224.183','172.10.224.132' ...
- 编译boost asio http/server 方法
这段时间学习boost 的asio 编程,想编译asio自带的http/server的程序,无奈在网上根本找不到方法,只能自己摸索学习. 登陆boost asio 的example 目录,(我 boo ...
- boost::asio 连接管理11 如何关闭连接
在实际产品运行中,对连接管理有了更新的认识,这里分享一下. shared_ptr管理连接对象的生命周期 shared_ptr的引用计数器决定了连接对象的生命周期.这里我说的连接对象就是在我的前文:ht ...
- Boost.Asio c++ 网络编程翻译(11)
*_at方法 这些方法在一个流上面做随机存取操作.你来指定read和write操作从什么地方開始(offset): async_read_at(stream, offset, buffer [, co ...
- 艰难的 debug 经历,vscode 无法获取远程环境 ssh 报错,windows 11 ssh
背景介绍 要做系统结构实验,学校和华为云合作使用华为云的 aarch64 裸机,需要使用 ssh 远程开发,笔者为了追求良好的开发体验,决定使用 vscode 开发,实验环境配置过程中遇到了两个问题, ...
- Boost.Asio基本原理(CSDN也有Markdown了,好开森)
Boost.Asio基本原理 这一章涵盖了使用Boost.Asio时必须知道的一些事情.我们也将深入研究比同步编程更复杂.更有乐趣的异步编程. 网络API 这一部分包含了当使用Boost.Asio编写 ...
- boost::asio 的同、异步方式
转自:http://blog.csdn.net/zhuky/archive/2010/03/10/5364574.aspx Boost.Asio是一个跨平台的网络及底层IO的C++编程库,它使用现代C ...
- boost::asio译文
Christopher Kohlhoff Copyright © 2003-2012 Christopher M. Kohlhoff 以Boost1.0的软件授权进行发布(见附带的LICENS ...
- PHP下通过file_get_contents\curl的方法实现获取远程网页内容(别忘了还有PhpRPC)
[php]PHP中file_get_contents()与file_put_contents()函数细节详解 php函数file_get_contents(一) 案例: 早在2010年时候遇到过这样的 ...
- boost asio 异步实现tcp通讯
---恢复内容开始--- asioboost 目录(?)[-] 一前言 二实现思路 通讯包数据结构 连接对象 连接管理器 服务器端的实现 对象串行化 一.前言 boost asio可算是一个简 ...
随机推荐
- 从 Rancher 学习 K8s - Rancher 的基础使用
舞台环境 Rancher v2.6.9 K8s v1.24.16 如果你不具备该环境,可以移步 使用 Rancher 安装 K8s 集群 查看.本文适合已经了解了 k8s 基础概念,想使用 ranch ...
- ABAP使用异步远程RFC实现并行处理
1.使用场景 当开发复杂报表,需要处理大量数据,不管怎么优化计算和查询语句,程序的运行效率还是达不到用户要求,怎么办? 为了解决这个问题,就需要程序实现并行处理. 本文档就是通过异步调用远程RFC的办 ...
- AcWing 第 12 场周赛
题目链接:Here AcWing 3805. 环形数组 签到题,循环减少出现次数,如果是 cnt[x] = 1 的话加入新的数组中 const int N = 1e3 + 10; int cnt[N] ...
- lighthouse性能优化分析工具使用
- svg组件封装
svg图标优点 文件体积小,能够被大量的压缩 图片可无限放大而不失真(矢量图的基本特征) 在视网膜显示屏上效果极佳 能够实现互动和滤镜效果 svg图标使用 1.安装相应的npm包: yarn add ...
- C#查找算法2:插值查找
插值查找,有序表的一种查找方式.插值查找是根据查找关键字与查找表中最大最小记录关键字比较后的查找方法.插值查找基于二分查找,将查找点的选择改进为自适应选择,提高查找效率. 原理: (midInd ...
- Vue第一篇 ES6的常用语法
01-变量的定义 <!DOCTYPE html> <html lang="en"> <head> <meta charset=" ...
- 每天学五分钟 Liunx 101 | 存储篇:LVM
LVM LVM(Logical Volume Manager),逻辑卷管理器.一种高级文件系统管理方式,它可以动态扩展文件系统. LVM 的示意图如下所示:
- PMP-干系人管理
转载请注明出处: 1.分析干系人管理的两大工具 1.1.权力-利益方阵 第一象限:严防死守(重点管理) 第二象限:投其所好(令其满意) 第三象限:保存 ...
- Python追踪内存占用
技术背景 当我们需要对python代码所占用的内存进行管理时,首先就需要有一个工具可以对当前的内存占用情况进行一个追踪.虽然在Top界面或者一些异步的工具中也能够看到实时的内存变化,还有一些工具可以统 ...