具有泛型语法的反射在重写方法的返回参数上失败

本文关键字:返回 方法 参数 失败 重写 泛型 语法 反射 | 更新日期: 2023-09-27 17:59:36

在搜索已知类型的属性时,为了避免老式的非泛型语法,通常使用System.Reflection.CustomAttributeExtensions类中的扩展方法(自.NET 4.5以来)。

但是,如果在重写方法(或重写属性/索引器的访问器)的返回参数上搜索属性,则此操作似乎会失败。

我正在经历这一切。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);
    }
  }
}

代码看起来可能"太长而无法读取",但它实际上只是一个重写的方法,其返回参数上有一个属性,并且显然是试图检索该属性实例。

堆栈跟踪:

未处理的异常:系统。IndexOutOfRangeException:索引超出了数组的界限。在系统中。属性GetParentDefinition(ParameterInfo param)在系统中。属性InternalParamGetCustomAttributes(ParameterInfo param,类型类型,布尔继承)在系统中。属性GetCustomAttributes(ParameterInfo元素、Type attributeType、Boolean inherit)在系统中。属性GetCustomAttribute(ParameterInfo元素、Type attributeType、Boolean inherit)在系统中。反射CustomAttributeExtensions。GetCustomAttribute[T](ParameterInfo元素)ReflectionTrouble。程序c:''MyPath''Program.cs:line 38中的Main()

我是不是做错了什么

这是一个bug吗?如果是,它是众所周知的吗?它是一个旧bug吗

具有泛型语法的反射在重写方法的返回参数上失败

这看起来确实像一个bug。问题似乎出在继承上。这确实有效:

ReturnParameter.GetCustomAttribute<MyMark>(inherit: false)

检索属性有两个工作方式稍有不同的代码路径:MemberInfo.GetCustomAttribute(较旧)和Attribute.GetCustomAttribute(较新且推荐使用)。还有一些通用的扩展方法,它们使用后一种更新的方法。它们之间的区别实际上在于它们处理继承的方式。NET 1.0忽略了属性、事件和参数的inherit参数。因此,为了不破坏事物,我们在中介绍了Attribute上的静态方法。NET 2.0(以及此错误)。

在向上继承树时,他们似乎忽略了特殊情况下的返回值参数(请参见此处)。您可以在GitHub repo中打开问题或Connect错误。