3

我有一个 COM 对象,其接口包含许多按以下方式定义的属性:

[propget] HRESULT Width([out, retval] LONG *lValue);

要从 C++ 访问此类属性,我需要添加如下代码:

LONG lValue;  
HRESULT hr = pInterface->get_Width(&lValue);  
if (FAILED(hr)) lValue = DEFAULT_VALUE;

这个块不是太长,但是当使用许多属性时,代码变得不太好看。有没有办法将属性访问代码分成一些宏或模板函数,以便能够直接使用属性,如下所示:

printf("The width of the object is %d", GET_OBJECT_PROPERTY(pInterace, Width, DEFAULT_VALUE));

UPD:VC2008编译器用于构建项目

UPD:谢谢大家!这是我的解决方案:

template <class interface_type, class property_type>
property_type GetPropertyValue(interface_type* pInterface, HRESULT(STDMETHODCALLTYPE interface_type::*pFunc)(property_type*), property_type DefaultValue = 0)
{
  property_type lValue;

  HRESULT hr = (*pInterface.*pFunc)(&lValue);

  if (FAILED(hr)) 
    lValue = DefaultValue;
  return lValue;
}

可以称为

LONG lVideoStreamCount = GetPropertyValue(pInfo, &IInterfaceName::get_VideoStreamCount);

我仍在寻找一种方法来从通话中消除此“IInterfaceName::”部分。

4

1 回答 1

0

#import "myInterface.tlb"

(正如 Hans 已经在评论中提出的那样。真的。我发现 C++ 中的 COM 无法忍受)

这会生成 - 在默认设置中 - 一个高级包装器。像您的示例这样的属性通过__decspec(property)公开,这是一个特定于编译器的扩展:

LONG lValue = pInterface->Width;

[编辑] 更多信息:

为了编译 .idl,我禁用了 MIDL 的 C 标头生成(如果需要,代理/存根生成除外)。

使用默认的#import 设置:

  • 你会得到高级包装器
    • 将 FAILED(hr) 变成_com_error异常
    • 变成[out, retval]真正的返回值,
    • 像属性一样暴露属性
    • raw_Method()通过等方式访问“原始” COM 接口 get_Property()
  • GUID 可通过 获得__uuidof(),例如,__uuidof(IMyInterface)而不是IID_IMyInterface
  • 所有定义都在基于 COM 库名称的命名空间中

但是,所有这些都可以通过#import 属性进行配置。

我将我的 .idl 放在一个单独的项目中,只生成 .tlb,我使用 #imported 标头来实现和使用接口。这意味着您必须对向导生成的代码进行一些调整,例如为 COM 方法实现添加前缀raw_

于 2012-04-27T12:41:42.927 回答