- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在使用 boost::interprocess 在进程之间共享对象。我有两个文件,一个生成结构对象并将该对象传递到具有 int 索引的映射中的“server.cpp”;和一个“client.cpp”文件,它检索内存数据并遍历数据,输出到控制台。
结构看起来像这样:
struct mydata o {
string MY_STRING;
int MY_INT;
};
和对象:
mydata o;
o.MY_STRING = "hello";
o.MY_INT = 45;
服务器和客户端都能正确编译。但是出于某种原因,如果我尝试访问客户端中的字符串而不是 float 或整数,客户端可执行文件会抛出段错误。例如下面的 second.MY_INT 将输出到控制台,但 second.MY_STRING 在运行时抛出此错误。
#include <boost/interprocess/managed_shared_memory.hpp>
#include <boost/interprocess/containers/map.hpp>
#include <boost/interprocess/allocators/allocator.hpp>
#include <functional>
#include <utility>
#include <iostream>
#include <string>
#include "objects.cpp" //definitions for objects
using std::string;
using namespace boost::interprocess;
int main ()
{
try
{
managed_shared_memory segment(open_or_create, "SharedMemoryName",65536);
//Again the map<Key, MappedType>'s value_type is std::pair<const Key, MappedType>, so the allocator must allocate that pair.
typedef int KeyType;
typedef order MappedType;
typedef std::pair<const int, order> ValueType;
//Assign allocator
typedef allocator<ValueType, managed_shared_memory::segment_manager> ShmemAllocator;
//The map
typedef map<KeyType, MappedType, std::less<KeyType>, ShmemAllocator> MySHMMap;
//Initialize the shared memory STL-compatible allocator
ShmemAllocator alloc_inst (segment.get_segment_manager());
//access the map in SHM through the offset ptr
MySHMMap :: iterator iter;
offset_ptr<MySHMMap> m_pmap = segment.find<MySHMMap>("MySHMMapName").first;
iter=m_pmap->begin();
for(; iter!=m_pmap->end();iter++)
{
//std::cout<<"\n "<<iter->first<<" "<<iter->second;
std::cout<<iter->first<<" "<<iter->second.MYINT<<" "<<iter->second.MYSTRING<<"\n";
}
}catch(std::exception &e)
{
std::cout<<" error " << e.what() <<std::endl;
shared_memory_object::remove("SharedMemoryName");
}
return 0;
}
运行时报错:
Segmentation fault (core dumped)
我很确定服务器正在将整个对象传递到内存,并且客户端可以检索它(因为我可以访问一些对象属性),这只是一个格式问题。
最佳答案
就像贾斯汀提到的,std::string
本身就是一个动态分配的容器。
仅使用 Interprocess 的 string
是不足够的。事实上,那只是boost::container::basic_string<>
真的。
重要的是使用分配器。
但是,使用带有分配器的映射,并在需要构建包含的容器时传递分配器(令人作呕)是很烦人的。
这样一来,您就不必知道分配器,任何知道如何使用作用域分配器的容器都会将分配器传递给嵌套容器。
#include <boost/interprocess/managed_shared_memory.hpp>
#include <boost/interprocess/containers/map.hpp>
#include <boost/interprocess/containers/string.hpp>
#include <boost/interprocess/allocators/allocator.hpp>
#include <boost/container/scoped_allocator.hpp>
#include <iostream>
#include <string>
namespace bip = boost::interprocess;
namespace Shared {
using Segment = bip::managed_shared_memory;
using Manager = Segment::segment_manager;
template <typename T> using Alloc
= boost::container::scoped_allocator_adaptor<bip::allocator<T, Manager> >;
using String = bip::basic_string<char, std::char_traits<char>, Alloc<char> >;
template <typename K, typename V, typename Cmp = std::less<K> > using Map
= bip::map<K, V, Cmp, Alloc<std::pair<K const, V> > >;
struct Order {
using allocator_type = Alloc<char>;
template <typename S, typename Alloc>
Order(int i, S const& s, Alloc alloc = {}) : i(i), s(s, alloc) {}
int i;
String s;
};
}
int main() {
try {
using namespace Shared;
Segment segment(bip::open_or_create, "095540a3-ceaa-4431-828d-df21d5e384ae", 65536);
auto& pmap = *segment.find_or_construct<Map<int, Order>>("MySHMMapName")(segment.get_segment_manager());
if (pmap.empty()) {
std::cout << "Inserting data\n";
auto insert = [&pmap](int i, auto&& s) {
using namespace std;
pmap.emplace(piecewise_construct, tie(i), tie(i, s));
};
insert(1, "one");
insert(2, "two");
insert(3, "three");
} else {
std::cout << "Existing data:\n";
for (auto& [k,v] : pmap) {
std::cout << k << " " << v.i << " " << v.s << "\n";
}
}
} catch (std::exception &e) {
std::cout << " error " << e.what() << std::endl;
bip::shared_memory_object::remove("095540a3-ceaa-4431-828d-df21d5e384ae");
}
}
我注意到 map 是 Map<int, Order>
: key 似乎复制了 Order
中的整数值.
Map<int, String>
你可以做到 Map<int, String>
并获得更流畅的体验(因为不需要 std::piecewise_construct
):
auto& pmap = *segment.find_or_construct<Map<int, String>>("MySHMMapName")(segment.get_segment_manager());
if (pmap.empty()) {
std::cout << "Inserting data\n";
pmap.emplace(1, "one");
pmap.emplace(2, "two");
pmap.emplace(3, "three");
}
或者,您应该考虑使用能够索引 Order
的多索引直接由类型的成员:
namespace bmi = boost::multi_index;
using Table = bmi::multi_index_container<Order,
bmi::indexed_by<
bmi::ordered_unique< bmi::member<Order, int, &Order::i> >
>,
Alloc<Order>
>;
遗憾的是,Multi Index 不能很好地处理使用分配器的嵌套类型,因此您必须再次传递它:
if (pmap.empty()) {
std::cout << "Inserting data\n";
pmap.emplace(1, "one", pmap.get_allocator());
pmap.emplace(2, "two", pmap.get_allocator());
pmap.emplace(3, "three", pmap.get_allocator());
} else {
std::cout << "Existing data:\n";
for (Order const& o : pmap) {
std::cout << o.i << " " << o.s << "\n";
}
// demonstrate lookup:
std::cout << "Finding element 2:" << pmap.find(2)->s << "\n";
}
打印
Existing data:
1 one
2 two
3 three
Finding element 2:two
¹ 在 Coliru 上使用映射文件。重构后的代码仅需一行更改。
关于c++ - boost::interprocess : cout a string variable when iterating through a map that references an object from a struct,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48675001/
我正在尝试表达以下内容: 给定一个矩阵和两个索引增量,返回矩阵中所有数字的四倍体:沿行,列或对角线的四倍体。 use std::iter::Iterator; use std::iter::Peeka
假设我们有以下类组成角色 Iterable : class Word-Char does Iterable { has @.words; method !pairize($item)
我编写了一个 ADT 排序二叉树,其功能如下: public Iterator getInorderIterator(){ return new InorderIterator(); } 有效
在包装(内部)迭代器时,通常必须将 __iter__ 方法重新路由到底层可迭代对象。考虑以下示例: class FancyNewClass(collections.Iterable): def
尽管如此,我遍历了以下 NSSet , NSMutableArray , NSFastEnumeration文档,我找不到下面提到的场景的令人满意的来源: 此处,NSMutableArray、NSAr
我发现在 Python 中 collections.Iterable 和 typing.Iterable 都可以用于类型注释和检查对象是否可迭代,即 >isinstance(obj, collecti
我想拆分实现 Iterator 的对象的输出分为两个实现 Iterator 的对象和 Iterator .由于其中一个输出的迭代次数可能比另一个多,因此我需要缓冲 Iterator 的输出。 (因为我
我正在尝试用 Rust 编写一个简单的迭代器: #[derive(Debug)] pub struct StackVec { storage: &'a mut [T], len: us
什么意思: Separator.Iterator.Element == Self.Iterator.Element.Iterator.Element 在this (Swift 标准库)swift 实例
调用 anIterable.iterator() 会返回新的迭代器还是现有的迭代器?它依赖于 Iterable 的实现吗? 更具体地说,以下代码是否按预期工作(即内部循环将从头开始迭代)? for (
我正在尝试转换 &str 的矢量对成一个 HashMap使用以下代码片段: use std::collections::HashMap; fn main() { let pairs = vec!(
这将使安全地迭代同一元素两次成为可能,或者为在项目类型中迭代的全局事物保持某种状态。 类似于: trait IterShort where Self: Borrow, { type I
我在 String 的字符上使用迭代器: pub fn is_yelling(message: &str) -> bool { let letters = message.chars().fi
这将使安全地迭代同一元素两次成为可能,或者为在项目类型中迭代的全局事物保持某种状态。 类似于: trait IterShort where Self: Borrow, { type I
要在 Rust 中实现迭代器,我们只需要实现 next 方法,如 in the documentation 所解释的那样.但是,Iterator 特征 has many more methods .
我正在为多个结构实现 Iterator 特性并遇到了一些问题。为什么为 Rows 实现 Iterator 显示错误?这是一个链接:link to playground 基本上为什么这不起作用? str
我将集合转储到磁盘上。当请求时,应该检索这些集合(没问题)和 iterator应该为它构建返回对检索到的值的引用。 iterator之后被丢弃了,我不再需要收藏了。我也希望它被删除。 到目前为止我尝试
我正在尝试为实现特征的结构实现默认迭代器。我的特征称为 DataRow,代表一行表格单元格,如下所示: pub trait DataRow { // Gets a cell by index
Rust 中是否有提供 iter() 的 Trait方法?我只找到了特征 IntoIterator ,供应into_iter() . 这里要明确一点:我不想要 Iterator特性,提供 next()
我想在迭代器上定义一个 .unique() 方法,使我能够在没有重复的情况下进行迭代。 use std::collections::HashSet; struct UniqueState {
我是一名优秀的程序员,十分优秀!