Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/csharp/336.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
C# 当反射仍然有效时,为什么动态方法调用失败?_C#_Dynamic_Reflection_Com - Fatal编程技术网

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);

尝试这两种方法,只使用第二个参数not
null
-
nto.Init(3,)
第二个参数是什么似乎无关紧要-NotImplementedException是相同的。根据文档,当第一个参数
lnInitType
为3(ADS\u NAME\u INITTYPE\u GC)时,该参数无论如何都会被忽略。有趣的是,没有一个
NameTranslate
方法可以通过
dynamic
调用。我认为,当输入某些参数时,在
NameTranslate
中的某个地方,调用了一些私有方法,甚至是
case
,但未实现该方法,并引发异常。