我不相信shared_memory_object::remove
如果附加了一个过程,文档中会提到任何会失败的内容。
请参阅此部分以供参考:删除共享内存。特别:
如果共享内存对象不存在或被另一个进程打开,此函数可能会失败。
这意味着无论如何调用都shared_memory_object::remove("foo")
将尝试删除名为“foo”的共享内存。
该函数的实现(此处的源代码)反映了该行为:
inline bool shared_memory_object::remove(const char *filename)
{
try{
//Make sure a temporary path is created for shared memory
std::string shmfile;
ipcdetail::tmp_filename(filename, shmfile);
return ipcdetail::delete_file(shmfile.c_str());
}
catch(...){
return false;
}
}
根据我发布生产代码的经验,我成功地在不再需要访问共享内存之前不调用。shared_memory_object::remove
我写了一个非常简单的示例主程序,您可能会发现它很有帮助。它将根据您的运行方式附加到、创建或删除共享内存。编译后,尝试以下步骤:
- 使用 c 运行以创建共享内存(默认为 1.0K)并插入虚拟数据
- 使用 o 运行以打开(“附加到”)共享内存并读取虚拟数据(默认情况下,每 10 秒循环读取一次)
- 在单独的会话中,使用 r 运行以删除共享内存
- 使用 o 再次运行以尝试打开。请注意,这将(几乎肯定)失败,因为在上一步中(再次,几乎肯定)删除了共享内存
- 随意从第二步终止该过程
至于为什么上面的第2步调用后仍然可以访问数据shared_memory_object::remove
,请参见构建托管共享内存。具体来说:
当我们打开一个托管共享内存
- 打开一个共享内存对象。
- 整个共享内存对象映射到进程的地址空间。
很可能,因为共享内存对象被映射到进程的地址空间,所以不再直接需要共享内存文件本身。
我意识到这是一个相当人为的例子,但我认为更具体的东西可能会有所帮助。
#include <cctype> // tolower()
#include <iostream>
#include <string>
#include <unistd.h> // sleep()
#include <boost/interprocess/shared_memory_object.hpp>
#include <boost/interprocess/managed_shared_memory.hpp>
int main(int argc, char *argv[])
{
using std::cerr; using std::cout; using std::endl;
using namespace boost::interprocess;
if (argc == 1) {
cout << "usage: " << argv[0] << " <command>\n 'c' create\n 'r' remove\n 'a' attach" << endl;
return 0;
}
const char * shm_name = "shared_memory_segment";
const char * data_name = "the_answer_to_everything";
switch (tolower(argv[1][0])) {
case 'c':
if (shared_memory_object::remove(shm_name)) { cout << "removed: " << shm_name << endl; }
managed_shared_memory(create_only, shm_name, 1024).construct<int>(data_name)(42);
cout << "created: " << shm_name << "\nadded int \"" << data_name << "\": " << 42 << endl;
break;
case 'r':
cout << (shared_memory_object::remove(shm_name) ? "removed: " : "failed to remove: " ) << shm_name << endl;
break;
case 'a':
{
managed_shared_memory segment(open_only, shm_name);
while (true) {
std::pair<int *, std::size_t> data = segment.find<int>( data_name );
if (!data.first || data.second == 0) {
cerr << "Allocation " << data_name << " either not found or empty" << endl;
break;
}
cout << "opened: " << shm_name << " (" << segment.get_segment_manager()->get_size()
<< " bytes)\nretrieved int \"" << data_name << "\": " << *data.first << endl;
sleep(10);
}
}
break;
default:
cerr << "unknown command" << endl;
break;
}
return 0;
}