为什么我不能使用Type.InvokeMember检索type out或ref参数的值?

时间:2012-03-26 22:53:50

标签: c# reflection reference-parameters

很长的标题,但我希望它具体。标题确实是个问题。即使InvokeMember正在调用的方法具有out参数并且正在为该参数赋值,但我无法获取该值。这是我最初使用的代码:

string parameter = "";
int result = Convert.ToInt32(typeof(Ability).InvokeMember(selectedMove, BindingFlags.InvokeMethod | BindingFlags.Public | BindingFlags.Static, null, null, new object[] { parameter }));

我改变了它,现在它按预期工作但我不知道为什么:

object[] args = new object[1];      //necessary to retrieve ref/out parameter
int result = Convert.ToInt32(typeof(Ability).InvokeMember(selectedMove, BindingFlags.InvokeMethod | BindingFlags.Public | BindingFlags.Static, null, null, args));

3 个答案:

答案 0 :(得分:6)

我只是想帮助那些正在努力(我做过)与非托管(COM)并获得ref-parameter的人。因此,当对COM方法使用InvokeMember时,您必须告诉哪些参数是ref-type。这是通过使用ParameterModifier-class实现的,例如:

object[] args = new object[3] { param1, param2, errorStr };
ParameterModifier pMod = new ParameterModifier(3);
pMod[2] = true;            
ParameterModifier[] mods = { pMod };

object tempObj = myCOMObject.GetType().InvokeMember("MyCOMMethod", BindingFlags.InvokeMethod | BindingFlags.NonPublic | BindingFlags.Public, null, myCOMObject, args, mods, null, null);

在上面的代码中,第3个参数被设置为引用(pMod [2] = true;)

答案 1 :(得分:5)

您的第二个代码段缺少相当重要的代码行。假设 out 参数的类型为string:

,它应该如下所示
object[] args = new object[1];      //necessary to retrieve ref/out parameter
int result = Convert.ToInt32(typeof(Ability).InvokeMember(selectedMove, 
    BindingFlags.InvokeMethod | BindingFlags.Public | BindingFlags.Static, 
    null, null, args));
string outValue = (string)args[0];  // <===  here!

现在也应该明白为什么你的第一个片段无法工作,你没有引用你传递的object []数组,所以你永远无法检索修改过的参数。

答案 2 :(得分:3)

在您的第一个代码示例中,对InvokeMember的调用不会修改parameter变量的值,它只是替换参数数组中的第一项(现在指向不同的{ {1}}实例)。由于您没有保留对此数组的引用,因此无法检索输出参数的值。

换句话说:数组最初包含string变量的副本(即对空字符串的引用的副本)。在调用之后,parameter和数组中的值引用了2个不同的字符串实例。