【问题标题】:Why does dynamic method invoke fail when reflection still works?为什么反射仍然有效时动态方法调用失败?
【发布时间】:2014-01-24 13:11:08
【问题描述】:

为什么 dynamic 对象不能在 NameTranslate COM 对象上调用这些方法,而反射可以?

使用动态的失败示例:

Type ntt = Type.GetTypeFromProgID("NameTranslate");
dynamic nto = Activator.CreateInstance(ntt);
nto.Init(3,null)

第三行失败,出现 NotImplementedException 和消息方法或操作未实现。

使用不同 COM 对象(WScript.ShellSendKeys)的类似尝试:

Type shellType = Type.GetTypeFromProgID("WScript.Shell");
dynamic shell = Activator.CreateInstance(shellType);
shell.SendKeys("abc");

回到第一个示例。如果我使用反射并使用InvokeMethod 方法调用方法,则一切正常。

使用反射的工作示例:

Type ntt = Type.GetTypeFromProgID("NameTranslate");
object nto = Activator.CreateInstance(ntt);
object[] initParams = new object[]{3,null};
ntt.InvokeMember("Init", BindingFlags.InvokeMethod, null, nto, initParams);

我相信这一定与 COM 对象的创建或标记方式有关 - 但在我的一生中,我在文档、对象浏览器或注册表中看不到任何指示这些 COM 对象及其子对象的内容/函数被标记为私有或其他通常会抛出 dynamic 关键字的东西。

MSDN 上的NameTranslate 文档:http://msdn.microsoft.com/en-us/library/windows/desktop/aa706046.aspx

【问题讨论】:

  • 尝试两种技术,只有第二个参数不是null - nto.Init(3,<somethig>)
  • 第二个参数是什么似乎并不重要——NotImplementedException 是一样的。根据文档,当第一个参数 lnInitType 为 3 (ADS_NAME_INITTYPE_GC) 时,无论如何都会忽略该参数。
  • 有趣的是,NameTranslate 方法都不能通过dynamic 调用。
  • 我想,发生的情况是,当输入某些参数时,NameTranslate 某处的某个私有方法甚至 case 被调用,但未实现并引发异常。

标签: c# dynamic reflection com


【解决方案1】:

有趣的是,NameTranslate 方法都不能通过dynamic 调用。下面我对此只有一个理论解释。

AFAIK,当 .NET DLR 为dynamic 调用处理 COM 对象时,它会尝试使用 COM 类型库(如果可用),然后求助于 IDispatch。这就是它与 Reflection 的不同之处,后者在与 COM 对象一起使用时会立即调用 IDispatch

ActiveDS 类型库 (C:\Windows\System32\activeds.tlb),用 OleView 观察,似乎有些不正确。它包含许多非自动化兼容的声明,包括接口:

interface IPrivateDispatch;
interface ITypeInfo;
interface ITypeComp;
interface ITypeLib;
interface IPrivateUnknown;

NameTranslate 本身的类定义如下所示:

[
  uuid(274FAE1F-3626-11D1-A3A4-00C04FB950DC)
]
coclass NameTranslate {
    [default] interface IADsNameTranslate;
    interface IDispatch;
};

coclass 中声明IDispatch 是不寻常的(尽管不被禁止)。

所以,我假设这种类型库和/或coclass 定义在这种情况下可能会混淆 DLR。

作为一种解决方法,您可以使用TlbImp.exe activeds.tlb 将其导入(这会产生一堆警告),将输出互操作程序集添加到您的项目并直接调用 API。这有效:

Type ntt = Type.GetTypeFromProgID("NameTranslate");
var nto = Activator.CreateInstance(ntt) as ActiveDs.IADsNameTranslate;
nto.Init(3, null);

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2019-08-07
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-09-09
    相关资源
    最近更新 更多