我使用boost :: interprocess :: managed_(windows_)shared_memory :: construct来构造一个包含自己类的进程间向量,它有一个类型为std :: string的成员变量,另一个类型为std :: vector,所以:
class myclass
{
public:
myclass()
{
}
std::string _mystring;
std::vector < int > _myintvector;
};
template < class _type >
struct typedefs
{
typedef boost::interprocess::managed_windows_shared_memory _memory;
typedef _memory::segment_manager _manager;
typedef boost::interprocess::allocator < _type, _manager > _allocator;
typedef boost::interprocess::vector < _type, _allocator > _vector;
};
typedef typedefs < myclass > tdmyclass;
int main ()
{
using namespace boost::interprocess;
managed_windows_shared_memory mem ( open_or_create, "mysharedmemory", 65536 );
tdmyclass::_vector * vec = mem.construct < tdmyclass::_vector > ( "mysharedvector" ) ( mem.get_segment_manager() );
myclass mytemp;
mytemp._mystring = "something";
mytemp._myintvector.push_back ( 100 );
mytemp._myintvector.push_back ( 200 );
vec->push_back ( mytemp );
/* waiting for the memory to be read is not what this is about,
so just imagine the programm stops here until everything we want to do is done */
}
我刚刚做了这个测试,我预计std :: string和std :: vector都没有工作,但是,如果我从另一个进程读取它,std :: string实际上有效,它包含我指定的字符串。这真让我感到惊讶。 另一边的std :: vector只能部分工作,size()返回的值是正确的,但是如果我想访问迭代器或者使用operator [],程序会崩溃。
所以,我的问题是,为什么会这样?我的意思是,我从来没有真正阅读过Visual Studio SDK的STL实现,但是std :: string只是一个std :: vector,其中有额外的函数适合字符串吗?难道他们都不使用std :: allocator - 这意味着BOTH std :: string和std :: vector在共享内存中不起作用?
谷歌搜索这并不会导致除了boost :: interprocess :: vector之外的任何东西,这不是我搜索的内容。所以我希望有人可以给我一些关于最新情况的细节^^
PS:如果我在上面的代码中输了一个拼写错误,请原谅我,我现在只是在这个页面编辑器中写了它,我有点太习惯了自动完成我的IDE ^^答案 0 :(得分:7)
std::string
有效,因为您的标准库实现使用小字符串优化(SSO)。这意味着字符串"something"
的值被复制到字符串对象本身,没有任何动态内存分配。因此,您可以从其他过程中读取它。对于较长的字符串(尝试30个字符),它将无法工作。
std::vector
不起作用,因为不允许按标准使用SSO。它在第一个进程的地址空间中分配内存,但是其他进程无法访问此内存。 .size()
有效,因为它将存储在内部向量本身,作为成员。
P.S。 string
和vector
之间存在许多差异。最重要的一个,也是最少提及的是string
is not a container from standard's point of view。