在附加进程时提升删除managed_shared_memory

时间:2013-08-08 16:03:56

标签: c++ boost ipc shared-memory

我有2个进程,进程1创建一个boost managed_shared_memory段,进程2打开这个段。然后重新启动过程1并且过程1的开始具有以下内容,

struct vshm_remove
{
    vshm_remove() 
    { 
        boost::interprocess::shared_memory_object::remove("VMySharedMemory"); 
    }
    ~vshm_remove()
    {
        boost::interprocess::shared_memory_object::remove("VMySharedMemory"); 
    }
} vremover;

我理解当进程1开始或结束时,将在我的共享内存上调用remove方法,但是如果没有附加进程2,它是否只能删除它?我使用以下内容附加到进程2中的共享内存,

boost::interprocess::managed_shared_memory *vfsegment;
vfsegment = new boost::interprocess::managed_shared_memory(boost::interprocess::open_only, "VMySharedMemory");

我注意到无论进程2是否已连接,都会删除共享内存。

2 个答案:

答案 0 :(得分:8)

我认为文档中没有提及shared_memory_object::remove如果附加了进程就会失败。

请参阅此部分以供参考:Removing shared memory。特别:

  
    

如果共享内存对象不存在或由另一个进程打开,则此函数可以失败。

  

这意味着,对shared_memory_object::remove("foo")的调用将尝试删除名为“foo”的共享内存,无论如何。

该功能的实现(source here)反映了这种行为:

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,直到我不再需要访问共享内存。

我写了一个非常简单的主程序示例,您可能会觉得有用。它将附加,创建或删除共享内存,具体取决于您运行它的方式。编译后,请尝试以下步骤:

  1. 使用c运行以创建共享内存(默认为1.0K)并插入虚拟数据
  2. 使用o运行打开(“附加到”)共享内存并读取虚拟数据(默认情况下,每10秒钟会读取一次循环)
  3. 在单独的会话中,使用r运行以删除共享内存
  4. 再次使用o运行尝试打开。请注意,这将(几乎可以肯定)失败,因为在上一步中(再次,几乎肯定)删除了共享内存
  5. 随意从第二步中删除该过程
  6. 为什么在调用shared_memory_object::remove后上面的第2步仍然可以访问数据,请参阅Constructing Managed Shared Memory。具体做法是:

      

    当我们打开托管共享内存时

         
        
    • 打开共享内存对象。
    •   
    • 整个共享内存对象映射到进程的“地址空间。
    • ”   

    最有可能的是,因为共享内存对象被映射到进程的地址空间,所以不再直接需要共享内存文件。

    我意识到这是一个相当人为的例子,但我认为更具体的一些可能会有所帮助。

    #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;
    }
    

答案 1 :(得分:0)

另外一个有趣的事情 - 再增加一个案例:

case 'w':
  {
    managed_shared_memory segment(open_only, shm_name);
      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;
      }
      *data.first = 17;
      cout << "opened: " << shm_name << " (" << segment.get_segment_manager()->get_size()
           << " bytes)\nretrieved int \"" << data_name << "\": " << *data.first << endl;
  }
  break;

aditional选项'w'导致内存被附加并写为'17'(“最随机的随机数”)。有了这个,您可以执行以下操作:

控制台1:执行'c',然后执行'a'。报告使用值42创建的内存。

控制台2:做'w'。在Console1上,您将看到该号码已更改。

控制台2:做'r'。内存已成功删除,控制台1仍然打印17。

控制台2:做'c'。它将报告使用值42创建的内存。

控制台2:做'a'。你会看到42,控制台1仍然打印17。

这确认 - 只要它在所有平台上以相同的方式工作,但是boost声明它确实 - 你可以用这种方式将内存块从一个进程发送到另一个进程,而“生产者”只需要确认“消费者”附加了块,以便“生产者”现在可以删除它。在连接下一个块之前,消费者也不必分离先前的块。