C# 当反射仍然有效时,为什么动态方法调用失败?
为什么当反射可以调用时,C# 当反射仍然有效时,为什么动态方法调用失败?,c#,dynamic,reflection,com,C#,Dynamic,Reflection,Com,为什么当反射可以调用时,动态对象不能调用NameTranslate COM对象上的这些方法 使用动态的失败示例: Type ntt = Type.GetTypeFromProgID("NameTranslate"); dynamic nto = Activator.CreateInstance(ntt); nto.Init(3,null) Type shellType = Type.GetTypeFromProgID("WScript.Shell"); dynamic shell = Acti
动态
对象不能调用NameTranslate COM对象上的这些方法
使用动态的失败示例:
Type ntt = Type.GetTypeFromProgID("NameTranslate");
dynamic nto = Activator.CreateInstance(ntt);
nto.Init(3,null)
Type shellType = Type.GetTypeFromProgID("WScript.Shell");
dynamic shell = Activator.CreateInstance(shellType);
shell.SendKeys("abc");
Type ntt = Type.GetTypeFromProgID("NameTranslate");
object nto = Activator.CreateInstance(ntt);
object[] initParams = new object[]{3,null};
ntt.InvokeMember("Init", BindingFlags.InvokeMethod, null, nto, initParams);
第三行失败,出现NotImplementedException,并显示方法或操作未实现的消息
使用不同COM对象(和)的类似尝试:
Type ntt = Type.GetTypeFromProgID("NameTranslate");
dynamic nto = Activator.CreateInstance(ntt);
nto.Init(3,null)
Type shellType = Type.GetTypeFromProgID("WScript.Shell");
dynamic shell = Activator.CreateInstance(shellType);
shell.SendKeys("abc");
Type ntt = Type.GetTypeFromProgID("NameTranslate");
object nto = Activator.CreateInstance(ntt);
object[] initParams = new object[]{3,null};
ntt.InvokeMember("Init", BindingFlags.InvokeMethod, null, nto, initParams);
回到第一个样本。如果我使用反射并调用使用该方法的方法,一切都会很好
使用反射的工作示例:
Type ntt = Type.GetTypeFromProgID("NameTranslate");
dynamic nto = Activator.CreateInstance(ntt);
nto.Init(3,null)
Type shellType = Type.GetTypeFromProgID("WScript.Shell");
dynamic shell = Activator.CreateInstance(shellType);
shell.SendKeys("abc");
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对象及其子对象/函数被标记为私有,或者其他通常会去掉动态
关键字的东西
MSDN上的NameTranslate文档:有趣的是,
NameTranslate
方法都不能通过dynamic
调用。下面,我只有一个理论上的解释
另外,当.NET DLR为动态调用处理COM对象时,它会尝试使用COM类型库(如果可用),然后求助于IDispatch
。这就是它与反射的不同之处,反射与COM对象一起使用时会立即调用IDispatch
用OleView查看ActiveDS类型库(C:\Windows\System32\ActiveDS.tlb
),似乎有点格式不正确。它包括许多非自动化兼容的声明,包括接口:
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);
尝试这两种方法,只使用第二个参数notnull
-nto.Init(3,)
第二个参数是什么似乎无关紧要-NotImplementedException是相同的。根据文档,当第一个参数lnInitType
为3(ADS\u NAME\u INITTYPE\u GC)时,该参数无论如何都会被忽略。有趣的是,没有一个NameTranslate
方法可以通过dynamic
调用。我认为,当输入某些参数时,在NameTranslate
中的某个地方,调用了一些私有方法,甚至是case
,但未实现该方法,并引发异常。