从存储在 CTreeCtrl 的 LPARAM 中的结构中检索 address/pointer (IXMLDOMNode*) - 不工作
Retrieving address/pointer (IXMLDOMNode*) from struct stored in LPARAM of CTreeCtrl - not working
在自定义 CTreeCtrl
class 中,我将指向自定义结构的指针添加为 LPARAM
-> 包含指向 IXMLDOMNode*
.[=23= 的地址]
稍后,我想从 LPARAM
获取指针并将其中的地址转换回 IXMLDOMNode*
。
这很好用,只要我在同一个 scope/function 中使用它(仅在功能中测试)。当我在另一个 function/class 中使用它来检索指针并将包含的地址转换为 IXMLDOMNode*
它是正确的 - 给定 - 地址。问题是,当我尝试使用它的一个函数时(例如 get_nodeName(BSTR *name)
),它会抛出一个未处理的异常。
Unhandled exception at 0x6522b1b8 in Program.exe: 0xC0000005: Access violation reading location 0x00000000.
插入:
HTREEITEM InsertIntoTree(CustomTreeCtrl &xCtrl, const HTREEITEM hCurrent, CComPtr<IXMLDOMNode> &cpCurrent)
{
TVINSERTSTRUCT tvItem = {0};
//[...]
//tvItem.item.mask has TVIF_PARAM && other flags set...
tvItem.item.lParam = TreeItemData(cpCurrent, 0).Alloc()->Address();
//[...]
}
获取(CustomTreeCtrl中的测试位置):
else
{
TreeItemData *tiData = TreeItemData::GetPointer(GetItemData(m_hActiveItem)); //Fine
CComPtr<IXMLDOMNode> F = tiData->GetCComPtr(); //Fine
BSTR Name = nullptr;
F->get_nodeName(&Name); //Unhandled Exception
}
结构:
struct TreeItemData
{
//[...]
TreeItemData(CComPtr<IXMLDOMNode> &pNode, DWORD dwFlags)
{
m_dwCComPtrAddress = reinterpret_cast<DWORD>(&*pNode);
m_dwFlags = dwFlags;
}
CComPtr<IXMLDOMNode> GetCComPtr(void)
{
return CComPtr<IXMLDOMNode>(reinterpret_cast<IXMLDOMNode*>(m_dwCComPtrAddress));
}
TreeItemData *Alloc(void)
{
return new TreeItemData(*this);
}
LPARAM Address(void)
{
return reinterpret_cast<LPARAM>(&*this);
}
static TreeItemData *GetPointer(DWORD Address)
{
return reinterpret_cast<TreeItemData*>(Address);
}
//[...]
DWORD m_dwCComPtrAddress;
DWORD m_dwFlags;
};
问:可能出了什么问题?我怎样才能让我的 - 工作 - 指向 IXMLDOMNode
的指针返回?
- 如果有指针,请始终使用 DWORD_PTR。只是为了 64 位兼容性。
- COM 指针使用引用计数。 CComPtr 关心这个引用计数。您将 CComPtr 传递给您的结构。您可以在不增加引用计数的情况下从 CComPtr 中获取对象指针。所以当你尝试使用它时,你的对象可能已经被销毁了,因为 CComPtr 已经被销毁了。
在您的 TreeItemData 结构中也使用智能指针。这也确保在释放树项结构时释放对象。
警告:当您从 CTreeCtrl 中删除树项目时,您知道必须释放所有这些结构。
在自定义 CTreeCtrl
class 中,我将指向自定义结构的指针添加为 LPARAM
-> 包含指向 IXMLDOMNode*
.[=23= 的地址]
稍后,我想从 LPARAM
获取指针并将其中的地址转换回 IXMLDOMNode*
。
这很好用,只要我在同一个 scope/function 中使用它(仅在功能中测试)。当我在另一个 function/class 中使用它来检索指针并将包含的地址转换为 IXMLDOMNode*
它是正确的 - 给定 - 地址。问题是,当我尝试使用它的一个函数时(例如 get_nodeName(BSTR *name)
),它会抛出一个未处理的异常。
Unhandled exception at 0x6522b1b8 in Program.exe: 0xC0000005: Access violation reading location 0x00000000.
插入:
HTREEITEM InsertIntoTree(CustomTreeCtrl &xCtrl, const HTREEITEM hCurrent, CComPtr<IXMLDOMNode> &cpCurrent)
{
TVINSERTSTRUCT tvItem = {0};
//[...]
//tvItem.item.mask has TVIF_PARAM && other flags set...
tvItem.item.lParam = TreeItemData(cpCurrent, 0).Alloc()->Address();
//[...]
}
获取(CustomTreeCtrl中的测试位置):
else
{
TreeItemData *tiData = TreeItemData::GetPointer(GetItemData(m_hActiveItem)); //Fine
CComPtr<IXMLDOMNode> F = tiData->GetCComPtr(); //Fine
BSTR Name = nullptr;
F->get_nodeName(&Name); //Unhandled Exception
}
结构:
struct TreeItemData
{
//[...]
TreeItemData(CComPtr<IXMLDOMNode> &pNode, DWORD dwFlags)
{
m_dwCComPtrAddress = reinterpret_cast<DWORD>(&*pNode);
m_dwFlags = dwFlags;
}
CComPtr<IXMLDOMNode> GetCComPtr(void)
{
return CComPtr<IXMLDOMNode>(reinterpret_cast<IXMLDOMNode*>(m_dwCComPtrAddress));
}
TreeItemData *Alloc(void)
{
return new TreeItemData(*this);
}
LPARAM Address(void)
{
return reinterpret_cast<LPARAM>(&*this);
}
static TreeItemData *GetPointer(DWORD Address)
{
return reinterpret_cast<TreeItemData*>(Address);
}
//[...]
DWORD m_dwCComPtrAddress;
DWORD m_dwFlags;
};
问:可能出了什么问题?我怎样才能让我的 - 工作 - 指向 IXMLDOMNode
的指针返回?
- 如果有指针,请始终使用 DWORD_PTR。只是为了 64 位兼容性。
- COM 指针使用引用计数。 CComPtr 关心这个引用计数。您将 CComPtr 传递给您的结构。您可以在不增加引用计数的情况下从 CComPtr 中获取对象指针。所以当你尝试使用它时,你的对象可能已经被销毁了,因为 CComPtr 已经被销毁了。
在您的 TreeItemData 结构中也使用智能指针。这也确保在释放树项结构时释放对象。
警告:当您从 CTreeCtrl 中删除树项目时,您知道必须释放所有这些结构。