我正在尝试创建一个管理(std)字符串的共享内存向量的类。
typedef boost::interprocess::allocator<std::string, boost::interprocess::managed_shared_memory::segment_manager> shmem_allocator;
typedef boost::interprocess::vector<std::string, shmem_allocator> shmem_vector;
shmem_mgr::shmem_mgr() :
shmem_(create_only, SHMEM_KEY, SHMEM_SIZE),
allocator_(shmem_.get_segment_manager())
{
mutex_ = shmem_.find_or_construct<interprocess_mutex>(SHMEM_MUTEX)();
condition_ = shmem_.find_or_construct<interprocess_condition>(SHMEM_CONDITION)();
//buffer_ is of type shmem_vector
buffer_ = shmem_.construct<shmem_vector>(SHMEM_BUFFER_KEY)(allocator_);
}
void shmem_mgr::run() {
running_ = true;
while(running_) {
scoped_lock<interprocess_mutex> lock ( *mutex_ );
int size = buffer_->size();
log_.debug() << size << " queued request(s) found" << std::endl; //LINE 27
for(int i=0; i<size; i++) {
log_.debug() << buffer_->at(i); // at() crashes my app
}
buffer_->clear(); //so does clear()
condition_->wait (lock);
}
}
客户端成功地向向量添加一个字符串(它也成功从缓冲区中读取该字符串以进行调试),管理器(上面的代码)接收信号(condtion变量),写入向量中有一个字符串(第27行),但是当它试图通过at()
获取该字符串时,应用程序崩溃了。
<小时/> 编辑:我已经发现,使用
std::string
是不可能的,在这种情况下,boost ipc中有一个string
容器。这并没有改变我需要(boost / std)字符串向量的事实......
问:如何在共享内存中传递字符串?我需要将它们存储在shmem中的某个缓冲区(一次能够存储&gt; 1)中,然后获取第二个进程 - 这是要求。输入始终为std::string
,输出也是如此,但shmem中的内部表示可能不同。
答案 0 :(得分:11)
来自docs。
#include <boost/interprocess/managed_shared_memory.hpp>
#include <boost/interprocess/containers/vector.hpp>
#include <boost/interprocess/containers/string.hpp>
#include <boost/interprocess/allocators/allocator.hpp>
int main ()
{
using namespace boost::interprocess;
//Typedefs
typedef allocator<char, managed_shared_memory::segment_manager>
CharAllocator;
typedef basic_string<char, std::char_traits<char>, CharAllocator>
MyShmString;
typedef allocator<MyShmString, managed_shared_memory::segment_manager>
StringAllocator;
typedef vector<MyShmString, StringAllocator>
MyShmStringVector;
//Open shared memory
//Remove shared memory on construction and destruction
struct shm_remove
{
shm_remove() { shared_memory_object::remove("MySharedMemory"); }
~shm_remove(){ shared_memory_object::remove("MySharedMemory"); }
} remover;
managed_shared_memory shm(create_only, "MySharedMemory", 10000);
//Create allocators
CharAllocator charallocator (shm.get_segment_manager());
StringAllocator stringallocator(shm.get_segment_manager());
//This string is in only in this process (the pointer pointing to the
//buffer that will hold the text is not in shared memory).
//But the buffer that will hold "this is my text" is allocated from
//shared memory
MyShmString mystring(charallocator);
mystring = "this is my text";
//This vector is only in this process (the pointer pointing to the
//buffer that will hold the MyShmString-s is not in shared memory).
//But the buffer that will hold 10 MyShmString-s is allocated from
//shared memory using StringAllocator. Since strings use a shared
//memory allocator (CharAllocator) the 10 buffers that hold
//"this is my text" text are also in shared memory.
MyShmStringVector myvector(stringallocator);
myvector.insert(myvector.begin(), 10, mystring);
//This vector is fully constructed in shared memory. All pointers
//buffers are constructed in the same shared memory segment
//This vector can be safely accessed from other processes.
MyShmStringVector *myshmvector =
shm.construct<MyShmStringVector>("myshmvector")(stringallocator);
myshmvector->insert(myshmvector->begin(), 10, mystring);
//Destroy vector. This will free all strings that the vector contains
shm.destroy_ptr(myshmvector);
return 0;
}
答案 1 :(得分:1)
您需要为可共享的stl类创建自定义分配器。你需要在分配器中定义一个基于自我的指针(ACE&amp; boost有这些)。在相反的一侧,(CONTIGUOUS)共享存储器通常驻留在不同的地址。你也需要一个共享内存分配子系统(堆管理器)(分配器分配) - 所有非平凡的低级代码,但绝大多数都是可行的,一旦你拥有它,它就可以在任何地方使用。如果你做了所有这些,你只需要传递非平面结构的位移(从(CONTIGUOUS !!)堆区域的开始)。
您可以创建队列以及您可能想要的所有其他内容 - 只要对象中的“指针”是基于自身的,并且非平面部分中的不连续部分来自一个大的连续部分。
你不能使用std :: string,因为除非你控制分配,否则标准字符串中的内存与你的共享内存没有任何关系 - 对于任何其他stl结构都是如此
解决/同意所有权问题也是必要的(像往常一样)
答案 2 :(得分:1)
您可以使用boost :: interprocess :: managed_shared_memory。以下程序在两个进程之间传递boost :: interprocess :: string。在我的机器上工作正常(Ubuntu Linux)。 您可以使用managed_shared_memory传递矢量或对象。 boost :: interprocess :: string有一个c_str()方法。
#include <boost/interprocess/managed_shared_memory.hpp>
#include <boost/interprocess/allocators/allocator.hpp>
#include <boost/interprocess/containers/string.hpp>
#include <cstring>
#include <cstdlib>
#include <string>
#include <iostream>
int main(int argc, char *argv[])
{
using namespace boost::interprocess;
typedef boost::interprocess::allocator<char, boost::interprocess::managed_shared_memory::segment_manager> CharAllocator;
typedef boost::interprocess::basic_string<char, std::char_traits<char>, CharAllocator> string;
if(argc == 1){ //Parent process
boost::interprocess::shared_memory_object::remove("MySharedMemory");
//Create a shared memory object.
managed_shared_memory shm (create_only, "MySharedMemory", 1024);
string *s = shm.find_or_construct<string>("String")("Hello!", shm.get_segment_manager());
std::cout << *s << std::endl;
//Launch child process
std::string s1(argv[0]); s1 += " child ";
if(0 != std::system(s1.c_str()))
return 1;
}
else{
//Open already created shared memory object.
managed_shared_memory shm (open_only, "MySharedMemory");
std::pair<string *,std::size_t> ret = shm.find<string>("String");
std::cout << *(ret.first) << std::endl;
}
return 0;
}