- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在尝试编写一个多线程 Web 服务器,但在将已接受的 boost::asio::ip::tcp::socket
对象从我的父线程传递到我的 child 线程。
我目前的解决方案非常笨拙,因为我不认为我理解正确的面向类的设计。尽管如此,在我继续前进并使其更清晰之前,我还是想让它与我笨拙的实现一起工作。
以下是出现问题的相关文件:
// consts_globs_shared.h
// User holds a username, and other string related info for this session
using boost::asio::ip::tcp;
std::map<
std::string,
std::deque<std::tuple<User, tcp::socket, std::deque<std::string>>
> chan_newusers;
std::mutex newusers_lock;
.
// server.cpp
#include "consts_globs_shared.h"
...
tcp::socket sock(io_service);
acceptor.accept(sock); // acceptor defined prior
...
std::unique_lock<std::mutex> lck(newuser_lock);
if (!chan_newusers.count(channel))
chan_newusers[channel]; // initialize
chan_newusers[channel].push_back(std::make_tuple(client,
std::move(sock),
temp_msgs));
.
// servlet.cpp
#include "consts_globs_shared.h"
...
for (auto it = chan_newusers[chan].begin(); it != chan_newusers[chan].end(); ++it) {
tcp::endpoint endpt = std::get<0>(*it).get_endpt(); // Users carry socket endpoint with them since that's copy constructible
tcp::socket temp(std::move(std::get<1>(*it)));
// have a class called servlet that holds a map called end_msgs
// @key: socket endpoints, @value: sockets
servlet.end_msgs[endpt]; // instantiate
servlet.end_msgs[endpt] = std::move(temp);
}
这个想法是服务器获取传入连接,将该套接字移动到全局 map 上,然后通知管理该聊天室的子线程有一个新用户需要获取。子线程从全局列表中抓取。我尝试省略不相关的代码,但我对如何实现这一点感到非常困惑。
我认为,当我将元组推送到映射中的双端队列时,元组会按值进行复制,因为我认为双端队列就是这样获取新元素的,而不是移动构造的。因此元组中的元素然后按值复制,并尝试按值复制套接字。但我不确定如何诊断是否发生了这种情况,也不确定如何修复它。
如果有任何建议,我将不胜感激。如果这篇文章可以变得更清晰或更容易理解,请告诉我。
编辑:这是我现在遇到的错误的一部分。这是巨大的,但我只会磨练看起来最相关的东西。
/usr/include/c++/5/ext/new_allocator.h:120:4: error: use of deleted
function ‘constexpr std::pair<_T1, _T2>::pair(const std::pair<_T1, _T2>&) [with _T1 = const
boost::asio::ip::basic_endpoint<boost::asio::ip::tcp>; _T2 =
boost::asio::basic_stream_socket<boost::asio::ip::tcp>]’
.
/usr/include/c++/5/ext/new_allocator.h:120:4: error: use of deleted
function ‘constexpr std::pair<_T1, _T2>::pair(const std::pair<_T1, _T2>&) [with _T1 = const
boost::asio::ip::basic_endpoint<boost::asio::ip::tcp>; _T2 =
boost::asio::basic_stream_socket<boost::asio::ip::tcp>]’
最佳答案
如果包含您遇到的错误消息会很有帮助,因为您的代码不是独立的。
I think that, when I make the tuple to be pushed onto the deque in the map, the tuple gets copied by value since that's how I believe deques grab new elements
你离目标不远
, rather than move constructed.
嗯。这取决于!如你can see push_back 在接受右值引用的 c++11 中获得了一个新的重载。如果您使用它,那么肯定会发生移动构造。
注意 c++11 also added emplace_back
哪个就地构建元组
我把你的样本做成独立的,有3个编译版本push_back或emplace_back信息:
auto& chan = chan_newusers[channel];
chan.emplace_back(client, std::move(sock), std::move(temp_msgs));
// // ALSO:
// chan.push_back(NewUserData(client, std::move(sock), std::move(temp_msgs)));
// // ALSO:
// chan.push_back({client, std::move(sock), std::move(temp_msgs)});
Note obviously, you can only move once, so uncomment only 1 of the three lines :)
#include <boost/asio.hpp>
#include <map>
#include <deque>
#include <tuple>
#include <string>
#include <mutex>
// User holds a username, and other string related info for this session
struct User {
std::string name;
// ...
};
// consts_globs_shared.h
using boost::asio::ip::tcp;
using ChannelId = std::string;
using NewUserData = std::tuple<User, tcp::socket, std::deque<std::string>>;
std::map<ChannelId, std::deque<NewUserData> > chan_newusers;
std::mutex newusers_lock;
struct SomethingInServerCpp {
boost::asio::io_service io_service;
void foo() {
// server.cpp
//...
tcp::socket sock(io_service);
tcp::acceptor acceptor(sock.get_io_service());
acceptor.bind({{}, 6767});
acceptor.accept(sock); // acceptor defined prior
//...
User client { "user1" };
std::string channel = "some_channel_from_the_request_I_guess";
std::deque<std::string> temp_msgs { "foo", "bar", "qux" };
std::unique_lock<std::mutex> lck(newusers_lock);
{
auto& chan = chan_newusers[channel];
chan.emplace_back(client, std::move(sock), std::move(temp_msgs));
// // ALSO:
// chan.push_back(NewUserData(client, std::move(sock), std::move(temp_msgs)));
// // ALSO:
// chan.push_back({client, std::move(sock), std::move(temp_msgs)});
}
}
};
int main() {
}
关于c++ - 使用 Boost Asio 套接字为双端队列调用移动构造函数时出现问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48254955/
为了让我的代码几乎完全用 Jquery 编写,我想用 Jquery 重写 AJAX 调用。 这是从网页到 Tomcat servlet 的调用。 我目前情况的类似代码: var http = new
我想使用 JNI 从 Java 调用 C 函数。在 C 函数中,我想创建一个 JVM 并调用一些 Java 对象。当我尝试创建 JVM 时,JNI_CreateJavaVM 返回 -1。 所以,我想知
环顾四周,我发现从 HTML 调用 Javascript 函数的最佳方法是将函数本身放在 HTML 中,而不是外部 Javascript 文件。所以我一直在网上四处寻找,找到了一些简短的教程,我可以根
我有这个组件: import {Component} from 'angular2/core'; import {UserServices} from '../services/UserService
我正在尝试用 C 实现一个简单的 OpenSSL 客户端/服务器模型,并且对 BIO_* 调用的使用感到好奇,与原始 SSL_* 调用相比,它允许一些不错的功能。 我对此比较陌生,所以我可能会完全错误
我正在处理有关异步调用的难题: 一个 JQuery 函数在用户点击时执行,然后调用一个 php 文件来检查用户输入是否与数据库中已有的信息重叠。如果是这样,则应提示用户确认是否要继续或取消,如果他单击
我有以下类(class)。 public Task { public static Task getInstance(String taskName) { return new
嘿,我正在构建一个小游戏,我正在通过制作一个数字 vector 来创建关卡,该数字 vector 通过枚举与 1-4 种颜色相关联。问题是循环(在 Simon::loadChallenge 中)我将颜
我有一个java spring boot api(数据接收器),客户端调用它来保存一些数据。一旦我完成了数据的持久化,我想进行另一个 api 调用(应该处理持久化的数据 - 数据聚合器),它应该自行异
首先,这涉及桌面应用程序而不是 ASP .Net 应用程序。 我已经为我的项目添加了一个 Web 引用,并构建了各种数据对象,例如 PayerInfo、Address 和 CreditCard。但问题
我如何告诉 FAKE 编译 .fs文件使用 fsc ? 解释如何传递参数的奖励积分,如 -a和 -target:dll . 编辑:我应该澄清一下,我正在尝试在没有 MSBuild/xbuild/.sl
我使用下划线模板配置了一个简单的主干模型和 View 。两个单独的 API 使用完全相同的配置。 API 1 按预期工作。 要重现该问题,请注释掉 API 1 的 URL,并取消注释 API 2 的
我不确定什么是更好的做法或更现实的做法。我希望从头开始创建目录系统,但不确定最佳方法是什么。 我想我在需要显示信息时使用对象,例如 info.php?id=100。有这样的代码用于显示 Game.cl
from datetime import timedelta class A: def __abs__(self): return -self class B1(A):
我在操作此生命游戏示例代码中的数组时遇到问题。 情况: “生命游戏”是约翰·康威发明的一种细胞自动化技术。它由一个细胞网格组成,这些细胞可以根据数学规则生存/死亡/繁殖。该网格中的活细胞和死细胞通过
如果我像这样调用 read() 来读取文件: unsigned char buf[512]; memset(buf, 0, sizeof(unsigned char) * 512); int fd;
我用 C 编写了一个简单的服务器,并希望调用它的功能与调用其他 C 守护程序的功能相同(例如使用 ./ftpd start 调用它并使用 ./ftpd stop 关闭该实例)。显然我遇到的问题是我不知
在 dos 中,当我粘贴此命令时它会起作用: "C:\Program Files (x86)\Google\Chrome\Application\chrome.exe" https://google.
在 dos 中,当我粘贴此命令时它会起作用: "C:\Program Files (x86)\Google\Chrome\Application\chrome.exe" https://google.
我希望能够从 cmd 在我的 Windows 10 计算机上调用 python3。 我已重新安装 Python3.7 以确保选择“添加到路径”选项,但仍无法调用 python3 并使 CMD 启动 P
我是一名优秀的程序员,十分优秀!