Какие атрибуты в.NET?

Что такое атрибуты в.NET, для чего они хороши и как мне создавать свои собственные атрибуты?

11 ответов

Решение

Метаданные. Данные о ваших объектах / методах / свойствах.

Например, я мог бы объявить Атрибут с именем: DisplayOrder, чтобы я мог легко контролировать, в каком порядке должны отображаться свойства в пользовательском интерфейсе. Затем я мог бы добавить его к классу и написать некоторые компоненты графического интерфейса, которые извлекают атрибуты и упорядочивают элементы интерфейса соответствующим образом.

public class DisplayWrapper
{
    private UnderlyingClass underlyingObject;

    public DisplayWrapper(UnderlyingClass u)
    {
        underlyingObject = u;
    }

    [DisplayOrder(1)]
    public int SomeInt
    {
        get
        {
            return underlyingObject .SomeInt;
        }
    }

    [DisplayOrder(2)]
    public DateTime SomeDate
    {
        get
        {
            return underlyingObject .SomeDate;
        }
    }
}

Тем самым гарантируя, что SomeInt всегда отображается перед SomeDate при работе с моими пользовательскими компонентами графического интерфейса.

Однако вы увидите, что они чаще всего используются за пределами среды прямого кодирования. Например, Windows Designer широко использует их, поэтому знает, как обращаться с объектами, сделанными на заказ. Использование BrowsableAttribute примерно так:

[Browsable(false)]
public SomeCustomType DontShowThisInTheDesigner
{
    get{/*do something*/}
}

Указывает конструктору не указывать это в доступных свойствах в окне "Свойства", например, во время разработки.

Вы также можете использовать их для генерации кода, операций перед компиляцией (таких как Post-Sharp) или операций во время выполнения, таких как Reflection.Emit. Например, вы можете написать немного кода для профилирования, который будет прозрачно упакован при каждом вызове, который ваш код делает, и умножит его. Вы можете "отказаться" от времени через атрибут, который вы помещаете в определенные методы.

public void SomeProfilingMethod(MethodInfo targetMethod, object target, params object[] args)
{
    bool time = true;
    foreach (Attribute a in target.GetCustomAttributes())
    {
        if (a.GetType() is NoTimingAttribute)
        {
            time = false;
            break;
        }
    }
    if (time)
    {
        StopWatch stopWatch = new StopWatch();
        stopWatch.Start();
        targetMethod.Invoke(target, args);
        stopWatch.Stop();
        HandleTimingOutput(targetMethod, stopWatch.Duration);
    }
    else
    {
        targetMethod.Invoke(target, args);
    }
}

Объявить их легко, просто создайте класс, который наследуется от Attribute.

public class DisplayOrderAttribute : Attribute
{
    private int order;

    public DisplayOrderAttribute(int order)
    {
        this.order = order;
    }

    public int Order
    {
        get { return order; }
    }
}

И помните, что когда вы используете атрибут, вы можете опустить суффикс "атрибут", компилятор добавит его для вас.

ПРИМЕЧАНИЕ. Атрибуты сами по себе ничего не делают - должен быть какой-то другой код, который их использует. Иногда этот код был написан для вас, но иногда вы должны написать его самостоятельно. Например, компилятор C# заботится о некоторых, а некоторые фреймворки используют некоторые (например, NUnit ищет [TestFixture] в классе и [Test] в тестовом методе при загрузке сборки).
Поэтому при создании собственного настраиваемого атрибута следует учитывать, что он никак не повлияет на поведение вашего кода. Вам нужно написать другую часть, которая проверяет атрибуты (через отражение) и воздействует на них.

Многие люди ответили, но никто до сих пор не упомянул об этом...

Атрибуты интенсивно используются с отражением. Отражение уже довольно медленно.

Очень стоит пометить ваши пользовательские атрибуты как sealed классы, чтобы улучшить их производительность во время выполнения.

Также неплохо подумать, где было бы целесообразно использовать такой атрибут, и присвоить свой атрибут (!), Чтобы указать это через AttributeUsage, Список доступных атрибутов может вас удивить:

  • сборочный
  • модуль
  • Учебный класс
  • Struct
  • Enum
  • Конструктор
  • метод
  • Имущество
  • поле
  • Событие
  • Интерфейс
  • параметр
  • делегат
  • ReturnValue
  • GenericParameter
  • Все

Также здорово, что атрибут AttributeUsage является частью подписи атрибута AttributeUsage. Ого за круговые зависимости!

[AttributeUsageAttribute(AttributeTargets.Class, Inherited = true)]
public sealed class AttributeUsageAttribute : Attribute

Атрибуты являются своего рода метаданными для маркировки классов. Это часто используется в WinForms, например, чтобы скрыть элементы управления от панели инструментов, но может быть реализовано в вашем собственном приложении, чтобы позволить экземплярам разных классов вести себя особым образом.

Начните с создания атрибута:

[AttributeUsage(AttributeTargets.Class, AllowMultiple=false, Inherited=true)]
public class SortOrderAttribute : Attribute
{
    public int SortOrder { get; set; }

    public SortOrderAttribute(int sortOrder)
    {
        this.SortOrder = sortOrder;
    }
}

Все классы атрибутов должны иметь суффикс "Атрибут", чтобы быть действительным.
После того, как это сделано, создайте класс, который использует атрибут.

[SortOrder(23)]
public class MyClass
{
    public MyClass()
    {
    }
}

Теперь вы можете проверить определенный класс " SortOrderAttribute (если он есть), выполнив следующие действия:

public class MyInvestigatorClass
{
    public void InvestigateTheAttribute()
    {
        // Get the type object for the class that is using
        // the attribute.
        Type type = typeof(MyClass);

        // Get all custom attributes for the type.
        object[] attributes = type.GetCustomAttributes(
            typeof(SortOrderAttribute), true);

        // Now let's make sure that we got at least one attribute.
        if (attributes != null && attributes.Length > 0)
        {
            // Get the first attribute in the list of custom attributes
            // that is of the type "SortOrderAttribute". This should only
            // be one since we said "AllowMultiple=false".
            SortOrderAttribute attribute = 
                attributes[0] as SortOrderAttribute;

            // Now we can get the sort order for the class "MyClass".
            int sortOrder = attribute.SortOrder;
        }
    }
}

Если вы хотите узнать больше об этом, вы всегда можете проверить MSDN, который имеет довольно хорошее описание.
Я надеюсь, что это помогло вам!

Атрибуты похожи на метаданные, применяемые к классам, методам или сборкам.

Они хороши для любого количества вещей (визуализация отладчика, пометка вещей как устаревших, пометка вещей как сериализуемых, список бесконечен).

Создать свои собственные легко, как пирог. Начни здесь:

http://msdn.microsoft.com/en-us/library/sw480ze8(VS.71).aspx

В проекте, над которым я сейчас работаю, есть набор объектов пользовательского интерфейса различных типов и редактор для сборки этих объектов для создания страниц для использования в основном приложении, немного похоже на конструктор форм в DevStudio. Эти объекты существуют в их собственной сборке, и каждый объект является классом, производным от UserControl и имеет собственный атрибут. Этот атрибут определяется так:

[AttributeUsage (AttributeTargets::Class)]
public ref class ControlDescriptionAttribute : Attribute
{
public:
  ControlDescriptionAttribute (String ^name, String ^description) :
    _name (name),
    _description (description)
  {
  }

  property String ^Name
  {
    String ^get () { return _name; }
  }

  property String ^Description
  {
    String ^get () { return _description; }
  }

private:
  String
    ^ _name,
    ^ _description;
};

и я применяю это к классу как это:

[ControlDescription ("Pie Chart", "Displays a pie chart")]
public ref class PieControl sealed : UserControl
{
  // stuff
};

о чем говорили предыдущие постеры.

Чтобы использовать атрибут, редактор имеет Generic::List <Type> содержащий типы управления. Существует список, из которого пользователь может перетащить и перетащить на страницу, чтобы создать экземпляр элемента управления. Чтобы заполнить список, я получаю ControlDescriptionAttribute для контроля и заполните запись в списке:

// done for each control type
array <Object ^>
  // get all the custom attributes
  ^attributes = controltype->GetCustomAttributes (true);

Type
  // this is the one we're interested in
  ^attributetype = ECMMainPageDisplay::ControlDescriptionAttribute::typeid;

// iterate over the custom attributes
for each (Object ^attribute in attributes)
{
  if (attributetype->IsInstanceOfType (attribute))
  {
    ECMMainPageDisplay::ControlDescriptionAttribute
      ^description = safe_cast <ECMMainPageDisplay::ControlDescriptionAttribute ^> (attribute);

    // get the name and description and create an entry in the list
    ListViewItem
      ^item = gcnew ListViewItem (description->Name);

    item->Tag = controltype->Name;
    item->SubItems->Add (description->Description);

    mcontrols->Items->Add (item);
    break;
  }
}

Примечание: выше приведен C++/CLI, но его нетрудно преобразовать в C# (да, я знаю, C++/CLI - это мерзость, но с этим я должен работать:-()

Вы можете поставить атрибуты на большинство вещей, и есть целый ряд предопределенных атрибутов. Упомянутый выше редактор также ищет настраиваемые атрибуты свойств, которые описывают свойство и способы его редактирования.

Как только вы поймете всю идею, вы удивитесь, как раньше жили без них.

Атрибут - это класс, который содержит некоторую функциональность, которую вы можете применить к объектам в вашем коде. Чтобы создать его, создайте класс, который наследуется от System.Attribute.

Что касается того, для чего они хороши... их использование практически безгранично.

http://www.codeproject.com/KB/cs/dotnetattributes.aspx

Как уже говорилось, Атрибуты относительно легко создавать. Другая часть работы - создание кода, который его использует. В большинстве случаев вы будете использовать отражение во время выполнения, чтобы изменить поведение в зависимости от наличия атрибута или его свойств. Существуют также сценарии, в которых вы будете проверять атрибуты в скомпилированном коде для выполнения статического анализа. Например, параметры могут быть помечены как ненулевые, и инструмент анализа может использовать это как подсказку.

Использование атрибутов и знание соответствующих сценариев для их использования - основная часть работы.

Атрибуты - это, по сути, биты данных, которые вы хотите прикрепить к вашим типам (классы, методы, события, перечисления и т. Д.)

Идея состоит в том, что во время выполнения какой-то другой тип / инфраструктура / инструмент будет запрашивать у вашего типа информацию в атрибуте и действовать в соответствии с ним.

Так, например, Visual Studio может запрашивать атрибуты стороннего элемента управления, чтобы выяснить, какие свойства элемента управления должны отображаться на панели "Свойства" во время разработки.

Атрибуты могут также использоваться в Аспектно-ориентированном программировании для внедрения / манипулирования объектами во время выполнения на основе атрибутов, которые их украшают, и добавления проверки, ведения журналов и т. Д. К объектам без влияния на бизнес-логику объекта.

Чтобы начать создавать атрибут, откройте исходный файл C#, введите attribute и нажмите [TAB]. Это расширится до шаблона для нового атрибута.

Пользовательские атрибуты можно использовать как простой способ определения значений тегов в подклассах без необходимости много раз писать один и тот же код для каждого подкласса. Я натолкнулся на хороший лаконичный пример Джона Уотерса о том, как определять и использовать пользовательские атрибуты в вашем собственном коде.

Существует учебное пособие по http://msdn.microsoft.com/en-us/library/aa288454(VS.71).aspx

Атрибуты также обычно используются для аспектно-ориентированного программирования. Для примера этого проверьте проект PostSharp.

Другие вопросы по тегам