Especifique a classe base necessária para alvos de atributo .NET
-
19-09-2019 - |
Pergunta
Tentei criar um atributo .NET personalizado com o código abaixo, mas acidentalmente parou da subclasse. Isso gerou um erro de compilador facilmente fixo mostrado no comentário.
// results in compiler error CS0641: Attribute 'AttributeUsage' is
// only valid on classes derived from System.Attribute
[AttributeUsage(AttributeTargets.Class)]
internal class ToolDeclarationAttribute
{
internal ToolDeclarationAttribute()
{
}
}
Minha pergunta é como o compilador sabe o [AttributeUsage]
atributo só pode ser aplicado a uma subclasse de System.Attribute
? Usando o .NET Reflector, não vejo nada de especial no AttributeUsageAttribute
Declaração de classe em si. Infelizmente, esse pode ser apenas um caso especial gerado pelo próprio compilador.
[Serializable, ComVisible(true), AttributeUsage(AttributeTargets.Class, Inherited=true)]
public sealed class AttributeUsageAttribute : Attribute
{
...
Eu gostaria de poder especificar que meu atributo personalizado só pode ser colocado nas subclasses de uma classe específica (ou interface). Isso é possível?
Solução
I would like to be able to specify that my custom attribute can only be placed on subclasses of a particular class (or interface). Is this possible?
Actually, there is a way to do this for subclasses (but not interfaces) using protected
- see Restricting Attribute Usage. To reproduce the code (but not the discussion):
abstract class MyBase {
[AttributeUsage(AttributeTargets.Property)]
protected sealed class SpecialAttribute : Attribute {}
}
class ShouldBeValid : MyBase {
[Special] // works fine
public int Foo { get; set; }
}
class ShouldBeInvalid { // not a subclass of MyBase
[Special] // type or namespace not found
[MyBase.Special] // inaccessible due to protection level
public int Bar{ get; set; }
}
Outras dicas
AttributeUsageAttribute
is just a magic class (like Attribute
itself is). This is a built-in compiler rule, and you cannot do something like that for your own attributes.
With ReSharper you can use [JetBrains.Annotations.BaseTypeRequired(typeof(YouBaseType))]