使用通用语法的反射在重写方法的 return 参数上失败
Reflection with generic syntax fails on a return parameter of an overridden method
为了在搜索已知类型的属性时避免使用老式的非泛型语法,通常使用 System.Reflection.CustomAttributeExtensions
class(自 .NET 4.5 起)中的扩展方法。
但是,如果您在覆盖方法(或覆盖的 property/indexer 的访问器)的 return 参数 上搜索属性,这似乎会失败).
我在 .NET 4.6.1 中遇到了这个问题。
简单再现(完整):
using System;
using System.Reflection;
namespace ReflectionTrouble
{
class B
{
//[return: MyMark("In base class")] // uncommenting does not help
public virtual int M() => 0;
}
class C : B
{
[return: MyMark("In inheriting class")] // commenting away attribute does not help
public override int M() => -1;
}
[AttributeUsage(AttributeTargets.ReturnValue, AllowMultiple = false, Inherited = false)] // commenting away AttributeUsage does not help
sealed class MyMarkAttribute : Attribute
{
public string Descr { get; }
public MyMarkAttribute(string descr)
{
Descr = descr;
}
public override string ToString() => $"MyMark({Descr})";
}
static class Program
{
static void Main()
{
var derivedReturnVal = typeof(C).GetMethod("M").ReturnParameter;
// usual new generic syntax (extension method in System.Refelction namespace):
var attr = derivedReturnVal.GetCustomAttribute<MyMarkAttribute>(); // BLOWS UP HERE, System.IndexOutOfRangeException: Index was outside the bounds of the array.
Console.WriteLine(attr);
// old non-generic syntax without extension method works:
var attr2 = ((MyMarkAttribute[])(derivedReturnVal.GetCustomAttributes(typeof(MyMarkAttribute), false)))[0]; // OK
Console.WriteLine(attr2);
}
}
}
该代码可能看起来 "too long to read",但它实际上只是一个重写方法,其 return 参数上有一个属性,并且明显试图检索该属性实例。
堆栈跟踪:
Unhandled Exception: System.IndexOutOfRangeException: Index was outside the bounds of the array.
at System.Attribute.GetParentDefinition(ParameterInfo param)
at System.Attribute.InternalParamGetCustomAttributes(ParameterInfo param, Type type, Boolean inherit)
at System.Attribute.GetCustomAttributes(ParameterInfo element, Type attributeType, Boolean inherit)
at System.Attribute.GetCustomAttribute(ParameterInfo element, Type attributeType, Boolean inherit)
at System.Reflection.CustomAttributeExtensions.GetCustomAttribute[T](ParameterInfo element)
at ReflectionTrouble.Program.Main() in c:\MyPath\Program.cs:line 38
我做错了什么吗?
这是一个错误吗?如果是,它是否广为人知,是否是一个老错误?
这看起来确实像一个错误。问题似乎与继承有关。这确实有效:
ReturnParameter.GetCustomAttribute<MyMark>(inherit: false)
检索属性有两个代码路径,它们的工作方式略有不同:MemberInfo.GetCustomAttribute
(较旧)和 Attribute.GetCustomAttribute
(较新和推荐)。还有通用扩展方法,它们使用后一种较新的方法。它们之间的区别确实在于它们处理继承的方式。 .NET 1.0 忽略了属性、事件和参数的 inherit
参数。所以为了不破坏东西,我们在 .NET 2.0 中引入了 Attribute
上的静态方法(连同这个错误)。
似乎他们在沿继承树上升时忽略了 return 值参数的特殊情况(参见 here)。您可以在 GitHub 存储库或 Connect 错误中提出问题。
为了在搜索已知类型的属性时避免使用老式的非泛型语法,通常使用 System.Reflection.CustomAttributeExtensions
class(自 .NET 4.5 起)中的扩展方法。
但是,如果您在覆盖方法(或覆盖的 property/indexer 的访问器)的 return 参数 上搜索属性,这似乎会失败).
我在 .NET 4.6.1 中遇到了这个问题。
简单再现(完整):
using System;
using System.Reflection;
namespace ReflectionTrouble
{
class B
{
//[return: MyMark("In base class")] // uncommenting does not help
public virtual int M() => 0;
}
class C : B
{
[return: MyMark("In inheriting class")] // commenting away attribute does not help
public override int M() => -1;
}
[AttributeUsage(AttributeTargets.ReturnValue, AllowMultiple = false, Inherited = false)] // commenting away AttributeUsage does not help
sealed class MyMarkAttribute : Attribute
{
public string Descr { get; }
public MyMarkAttribute(string descr)
{
Descr = descr;
}
public override string ToString() => $"MyMark({Descr})";
}
static class Program
{
static void Main()
{
var derivedReturnVal = typeof(C).GetMethod("M").ReturnParameter;
// usual new generic syntax (extension method in System.Refelction namespace):
var attr = derivedReturnVal.GetCustomAttribute<MyMarkAttribute>(); // BLOWS UP HERE, System.IndexOutOfRangeException: Index was outside the bounds of the array.
Console.WriteLine(attr);
// old non-generic syntax without extension method works:
var attr2 = ((MyMarkAttribute[])(derivedReturnVal.GetCustomAttributes(typeof(MyMarkAttribute), false)))[0]; // OK
Console.WriteLine(attr2);
}
}
}
该代码可能看起来 "too long to read",但它实际上只是一个重写方法,其 return 参数上有一个属性,并且明显试图检索该属性实例。
堆栈跟踪:
Unhandled Exception: System.IndexOutOfRangeException: Index was outside the bounds of the array. at System.Attribute.GetParentDefinition(ParameterInfo param) at System.Attribute.InternalParamGetCustomAttributes(ParameterInfo param, Type type, Boolean inherit) at System.Attribute.GetCustomAttributes(ParameterInfo element, Type attributeType, Boolean inherit) at System.Attribute.GetCustomAttribute(ParameterInfo element, Type attributeType, Boolean inherit) at System.Reflection.CustomAttributeExtensions.GetCustomAttribute[T](ParameterInfo element) at ReflectionTrouble.Program.Main() in c:\MyPath\Program.cs:line 38
我做错了什么吗?
这是一个错误吗?如果是,它是否广为人知,是否是一个老错误?
这看起来确实像一个错误。问题似乎与继承有关。这确实有效:
ReturnParameter.GetCustomAttribute<MyMark>(inherit: false)
检索属性有两个代码路径,它们的工作方式略有不同:MemberInfo.GetCustomAttribute
(较旧)和 Attribute.GetCustomAttribute
(较新和推荐)。还有通用扩展方法,它们使用后一种较新的方法。它们之间的区别确实在于它们处理继承的方式。 .NET 1.0 忽略了属性、事件和参数的 inherit
参数。所以为了不破坏东西,我们在 .NET 2.0 中引入了 Attribute
上的静态方法(连同这个错误)。
似乎他们在沿继承树上升时忽略了 return 值参数的特殊情况(参见 here)。您可以在 GitHub 存储库或 Connect 错误中提出问题。