如何使用IDispatch.Invoke调用接收浮点值的函数?

How do I call functions that receive floating point values using IDispatch.Invoke?

本文关键字:函数 IDispatch 何使用 Invoke 调用      更新时间:2023-10-16

我在使用IDispatch.Invoke调用具有浮点参数和浮点结果的函数时遇到麻烦。

这是一个最小的复制:

#include <atlbase.h>
#include <comutil.h>
int main(int argc, char* argv[])
{
    CoInitialize(NULL);
    CComPtr<IDispatch> wordapp;
    if (SUCCEEDED(wordapp.CoCreateInstance(L"Word.Application", NULL, CLSCTX_LOCAL_SERVER)))
    {
        CComVariant result;
        CComVariant centimeters((float)2.0);
        CComVariant retval = wordapp.Invoke1(L"CentimetersToPoints", &centimeters, &result);
    }
    return 0;
}

我使用ATL CComPtr使事情更干净。但它是一个非常松散的包裹在IDispatch.Invoke

当我运行这个时,对Invoke1的调用失败并返回E_FAIL

我怀疑这个问题与使用浮点参数和/或返回值有关。如果我调用一个不使用这些值的函数,调用成功:

CComVariant retval = wordapp.Invoke0(L"ProductCode", &result);

我注意到,如果我从VBS或从PowerShell调用这个函数,它会成功。我假设它们都使用晚绑定IDispatch,因此这将表明我所尝试的至少是可能的。

那么,我如何使用IDispatch调用这个函数?

嗯,这是一个令人头疼的问题。这个"方法"的文档很容易误导人,它不是一个方法。它的行为更像是一个索引属性,并且需要DISPATCH_METHOD | DISPATCH_PROPERTYGET标志。CComPtr不支持这个,你需要手动旋转这个。下面的代码可以运行:

    CComVariant result;
    CComVariant centimeters((float)2.0);
    DISPID dispid;
    LPOLESTR name = L"CentimetersToPoints";
    HRESULT hr = wordapp->GetIDsOfNames(IID_NULL, &name, 1, LOCALE_USER_DEFAULT, &dispid);
    assert(SUCCEEDED(hr));
    DISPPARAMS dispparams = { &centimeters, NULL, 1, 0};
    hr = wordapp->Invoke(dispid, IID_NULL, LOCALE_USER_DEFAULT,
            DISPATCH_METHOD | DISPATCH_PROPERTYGET, &dispparams, &result, nullptr, nullptr);
    assert(SUCCEEDED(hr));