gpt4 book ai didi

使用命名共享内存的 C++ 问题

转载 作者:塔克拉玛干 更新时间:2023-11-03 07:07:44 25 4
gpt4 key购买 nike

我已经尝试实现一个共享内存接口(interface),但是我无法让它工作。

因为它还没有工作,我想寻求帮助。我的应用程序需要共享内存,这是一个在多个进程上运行的多目标进化算法,但是各个进程需要交换信息,而不是将它转储到物理文件中十亿次,我宁愿使用它共享内存方法。

我在 Win7x64 上使用 VS v120 中的 C++。为了测试,所有这些代码都发生在同一个进程中,直到我弄清楚为止。

我的文件名是一个常量字符串

 m_Filename = "Local\\shared_memory"

m_BufferSize = 1024

编辑 1:

所以我看到我在这里尝试做的事情有些困惑,我也很困惑。查看来自 MSDN 的官方文档,它使用带有 INVALID_HANDLE_VALUE 的文件映射,而且它们似乎没有在磁盘上创建文件。这对我的解决方案很好。我不需要磁盘上的文件,尽管两者都可以。我尝试用另一种方法来做的原因是因为第一种方法失败了,我开始搜索,我在这里遇到了一些线程,人们说他们也需要制作实际文件。

这是一个更完整的代码,是的,我确实检查了错误代码。

m_Filename 在类构造函数中设置。缓冲区大小是恒定的。我已经删除了执行物理文件内容的代码,我想实际上并不需要它?

    void MemoryMapper::_CreateMappedFile() {

m_Handle = CreateFileMapping(INVALID_HANDLE_VALUE, NULL, PAGE_READWRITE,
0, m_BufferSize, m_Filename.c_str());

if (m_Handle == NULL)
{
std::cout << m_DebugErrorTitle << "_CreateMappedFile(): " << MM_ERROR_CREATE_FAILED <<
" (" << GetLastError() << ")" << std::endl;
return;
}

m_pBuffer = (LPTSTR)MapViewOfFile(m_Handle, FILE_MAP_ALL_ACCESS, 0, 0, m_BufferSize);

if (m_pBuffer == NULL)
{
std::cout << m_DebugErrorTitle << "_CreateMappedFile(): " << MM_ERROR_MAPPING_FAILED <<
" (" << GetLastError() << ")" << std::endl;
CloseHandle(m_Handle);
return;
}

TCHAR szMsg[] = TEXT("Test message.");
CopyMemory((PVOID)m_pBuffer, szMsg, (_tcslen(szMsg) * sizeof(TCHAR)));

if (!UnmapViewOfFile(m_pBuffer)) {
std::cout << m_DebugErrorTitle << "_CreateMappedFile(): UnmapViewOfFile() returned false. (" << GetLastError() << ")" << std::endl;
}
if (!CloseHandle(m_Handle)) {
std::cout << m_DebugErrorTitle << "_CreateMappedFile(): CloseHandle() returned false. (" << GetLastError() << ")" << std::endl;
}

if (m_Debug > 1) { std::cout << m_DebugTitle << "Created mapped file: '" << m_Filename << "'." << std::endl; }
}

运行这段代码,我得到了控制台消息:[MemoryMapper] 创建映射文件:'Local\shared_memory'。

然后,在同一个进程中,为了测试??我尝试再次打开文件。这次我收到错误代码 2,表示该文件不存在。

bool MemoryMapper::_Open(const std::string& fn) {

if (m_Debug > 2) { std::cout << m_DebugTitle << "Open '" << fn << "'." << std::endl; }

m_OpenHandle = OpenFileMapping(FILE_MAP_ALL_ACCESS, FALSE, fn.c_str());

if (m_OpenHandle == NULL)
{
std::cout << m_DebugErrorTitle << " _Open('" << fn << "'): " << MM_ERROR_OPEN_FAILED << " (" << GetLastError() << ")" << std::endl;
m_IsOpen = false;
return m_IsOpen;
}
m_IsOpen = true;
if (m_Debug > 1) { std::cout << m_DebugTitle << "Open: " << std::to_string(m_IsOpen) << std::endl; }
return m_IsOpen;
}

文件名相同。 100%。

为什么我打不开文件?

此外,在创建共享内存对象之前,我是否应该检查是否存在具有设置文件名的共享内存对象?应用程序终止时对象是否被清除?

编辑 2:

在我想使用共享对象的期间,初始 CreateFileMapping() 给出的句柄似乎必须保持打开状态?

我尝试过这个,现在它似乎工作正常。我可以使用单独的调用创建对象、打开它、写入和关闭它。我的错误是在创建时关闭句柄,但这是正确的吗?

最佳答案

首先,如果你试图实现一个共享内存接口(interface)——你在磁盘上创建文件的目的是什么?你可以这样做,但在这种情况下绝对不需要

(3) Now the object in itself should exist in memory, no?

现在对象确实存在于 NT namespace 中,但直到您不关闭它的最后一个句柄。当最后一个句柄关闭时,对象名称将从 NT namespace 中删除,除非您不使用 OBJ_PERMANENT OBJECT_ATTRIBUTES 中标记.但为此需要使用 NtCreateSection相反 CreateFileMapping并且有 SE_CREATE_PERMANENT_PRIVILEGE

如果OpenFileMapping失败并出现错误 ERROR_FILE_NOT_FOUND这意味着您在调用中使用的名称不存在于 NT 命名空间中。这可能是由以下几个原因造成的: - 调用 OpenFileMapping 的进程在另一个终端 session 下运行(Local\shared_memory 扩展为 \Sessions\<SessionId>\BaseNamedObjects\shared_memory 或扩展为 \BaseNamedObjects\shared_memory 如果您从 session 0 运行它)。你可以简单地用名字弄错。

但最重要的是 - 您关闭由 CreateFileMapping 返回的部分句柄在你打电话之前 OpenFileMapping .因为你不使用 OBJ_PERMANENT关闭所有打开的句柄时删除对象名称。如果存在对它的引用,对象本身可以继续存在 - 比如当你调用 MapViewOfFile 时- 您创建了对部分的引用 - 在您不取消映射之前它不会被删除,但是当所有打开的句柄关闭时,部分名称无论如何都会被删除。


我如何假设开始 - OpenFileMapping失败,因为部分在它调用的时间不再存在。它处理已经关闭。 void MemoryMapper::_CreateMappedFile()实际上什么都不做——这个函数创建临时对象,对其进行一些操作并在退出时销毁。这一切在函数返回后没有任何影响

关于使用命名共享内存的 C++ 问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47021915/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com