gpt4 book ai didi

c++ - 如何使用boost::beast,下载文件无阻塞且带有响应

转载 作者:行者123 更新时间:2023-12-02 10:11:38 85 4
gpt4 key购买 nike

我从this example开始,所以不会发布所有代码。我的目标是下载大文件而不阻塞我的主线程。第二个目标是获取通知,以便我可以更新进度条。我的代码确实有两种工作方式。首先是只是ioc.run();并使其工作,我下载了文件。但是无论如何我都找不到没有阻塞就开始 session 的方法。
第二种方法,我可以将调用降低到http::async_read_some,并且该调用可以运行,但是我无法获得可以使用的响应。我不知道是否有一种方法可以传递捕获的lambda。#if 0..#else..#endif切换方法。我敢肯定有一个简单的方法,但是我看不到它。我将在工作时清理代码,例如设置本地文件名。谢谢。

    std::size_t on_read_some(boost::system::error_code ec, std::size_t bytes_transferred)
{
if (ec);//deal with it...
if (!bValidConnection) {
std::string_view view((const char*)buffer_.data().data(), bytes_transferred);
auto pos = view.find("Content-Length:");
if (pos == std::string_view::npos)
;//error
file_size = std::stoi(view.substr(pos+sizeof("Content-Length:")).data());
if (!file_size)
;//error
bValidConnection = true;
}
else {
file_pos += bytes_transferred;
response_call(ec, file_pos);
}
#if 0
std::cout << "in on_read_some caller\n";
http::async_read_some(stream_, buffer_, file_parser_, std::bind(
response_call,
std::placeholders::_1,
std::placeholders::_2));
#else
std::cout << "in on_read_some inner\n";
http::async_read_some(stream_, buffer_, file_parser_, std::bind(
&session::on_read_some,
shared_from_this(),
std::placeholders::_1,
std::placeholders::_2));
#endif
return buffer_.size();
}
主要的,困惑的.....
struct lambda_type {
bool bDone = false;
void operator ()(const boost::system::error_code ec, std::size_t bytes_transferred) {
;
}
};
int main(int argc, char** argv)
{
auto const host = "reserveanalyst.com";
auto const port = "443";
auto const target = "/downloads/demo.msi";
int version = argc == 5 && !std::strcmp("1.0", argv[4]) ? 10 : 11;

boost::asio::io_context ioc;
ssl::context ctx{ ssl::context::sslv23_client };

load_root_certificates(ctx);
//ctx.load_verify_file("ca.pem");

auto so = std::make_shared<session>(ioc, ctx);
so->run(host, port, target, version);

bool bDone = false;
auto const lambda = [](const boost::system::error_code ec, std::size_t bytes_transferred) {
std::cout << "data lambda bytes: " << bytes_transferred << " er: " << ec.message() << std::endl;
};

lambda_type lambda2;
so->set_response_call(lambda);
ioc.run();

std::cout << "not in ioc.run()!!!!!!!!" << std::endl;

so->async_read_some(lambda);

//pseudo message pump when working.........
for (;;) {
std::this_thread::sleep_for(250ms);
std::cout << "time" << std::endl;
}
return EXIT_SUCCESS;
}
还有我添加到 class session中的内容
class session : public std::enable_shared_from_this<session>
{
using response_call_type = void(*)(boost::system::error_code ec, std::size_t bytes_transferred);
http::response_parser<http::file_body> file_parser_;
response_call_type response_call;
//
bool bValidConnection = false;
std::size_t file_pos = 0;
std::size_t file_size = 0;

public:
auto& get_result() { return res_; }
auto& get_buffer() { return buffer_; }
void set_response_call(response_call_type the_call) { response_call = the_call; }

最佳答案

我强烈建议不要使用低级[async_]read_some函数,而不要按照http::[async_]read的意图使用http::response_parser<http::buffer_body>我确实有一个例子-由于它还使用Boost Process来同时解压缩主体数据,因此有点复杂,但是无论它如何显示如何使用它:
How to read data from Internet using muli-threading with connecting only once?
我想我可以在给出更完整的代码的情况下针对您的特定示例进行调整,但是上面的内容也许足够好?另请参阅我用作“灵感”的libs / beast / example / doc / http_examples.hpp中的“中继HTTP消息”。

Caution: the buffer arithmetic is not intuitive. I think this is unfortunate and should not have been necessary, so pay (very) close attention to these samples for exactly how that's done.

关于c++ - 如何使用boost::beast,下载文件无阻塞且带有响应,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63330104/

85 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com