Атрибуты на интерфейсе
-
05-07-2019 - |
Вопрос
У меня есть интерфейс, который определяет некоторые методы с атрибутами. Эти атрибуты должны быть доступны из вызывающего метода, но метод, который я имею, не извлекает атрибуты из интерфейса. Чего мне не хватает?
public class SomeClass: ISomeInterface
{
MyAttribute GetAttribute()
{
StackTrace stackTrace = new StackTrace();
StackFrame stackFrame = stackTrace.GetFrame(1);
MethodBase methodBase = stackFrame.GetMethod();
object[] attributes = methodBase.GetCustomAttributes(typeof(MyAttribute), true);
if (attributes.Count() == 0)
throw new Exception("could not find MyAttribute defined for " + methodBase.Name);
return attributes[0] as MyAttribute;
}
void DoSomething()
{
MyAttribute ma = GetAttribute();
string s = ma.SomeProperty;
}
}
Решение
methodBase будет методом класса, а не интерфейса. Вам нужно будет найти тот же метод в интерфейсе. В C # это немного проще (так как оно должно быть одноименным), но вам нужно учитывать такие вещи, как явная реализация. Если у вас есть код VB, это будет сложнее, так как метод VB " Foo " Можно реализовать интерфейсный метод «Бар». Для этого вам нужно изучить карту интерфейса:
using System;
using System.ComponentModel;
using System.Diagnostics;
using System.Reflection;
interface IFoo
{
void AAA(); // just to push Bar to index 1
[Description("abc")]
void Bar();
}
class Foo : IFoo
{
public void AAA() { } // just to satisfy interface
static void Main()
{
IFoo foo = new Foo();
foo.Bar();
}
void IFoo.Bar()
{
GetAttribute();
}
void GetAttribute()
{ // simplified just to obtain the [Description]
StackTrace stackTrace = new StackTrace();
StackFrame stackFrame = stackTrace.GetFrame(1);
MethodBase classMethod = stackFrame.GetMethod();
InterfaceMapping map = GetType().GetInterfaceMap(typeof(IFoo));
int index = Array.IndexOf(map.TargetMethods, classMethod);
MethodBase iMethod = map.InterfaceMethods[index];
string desc = ((DescriptionAttribute)Attribute.GetCustomAttribute(iMethod, typeof(DescriptionAttribute))).Description;
}
}
Другие советы
Метод Марка будет работать для неуниверсальных интерфейсов. Но, похоже, я имею дело с теми, у кого есть дженерики
interface IFoo<T> {}
class Foo<T>: IFoo<T>
{
T Bar()
}
Похоже, что T заменяется фактическим classType на карте. TargetMethods.
Хотя я сначала признаюсь, что никогда не пытался прикрепить атрибуты к интерфейсам, но что-то вроде следующей работы для вас?
public abstract class SomeBaseClass: ISomeInterface
{
[MyAttribute]
abstract void MyTestMethod();
}
public SomeClass : SomeBaseClass{
MyAttribute GetAttribute(){
Type t = GetType();
object[] attibutes = t.GetCustomAttributes(typeof(MyAttribute), false);
if (attributes.Count() == 0)
throw new Exception("could not find MyAttribute defined for " + methodBase.Name);
return attributes[0] as MyAttribute;
}
....
}