C # переопределяет атрибут в подклассе

#c# #attributes

#c# #атрибуты

Вопрос:

 public class MyWebControl {

    [ExternallyVisible]
    public string StyleString {get;set;}

}

public class SmarterWebControl : MyWebControl {

    [ExternallyVisible]
    public string CssName{get;set;}

    new public string StyleString {get;set;} //Doesn't work

}
  

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

Редактировать: Упс, похоже, я забыл скомпилировать или что-то в этом роде, потому что код, опубликованный выше, на самом деле работает!

Комментарии:

1. рад, что это работает, я был озадачен, почему этого не произойдет. Должен сказать, однако, я не уверен, что мне нравится использование new здесь (хотя я в некотором роде против new в целом). Для тех, кто смотрит на класс, это не говорит им, почему вы это делаете. С другой стороны, использование атрибута с параметром false является самодокументируемым imho

2. Верно, и если бы я писал все это с нуля, я бы, вероятно, сделал это таким образом 😉

Ответ №1:

Именно поэтому атрибуты фреймворка, которые могут быть «переопределены», принимают логический параметр, который (на первый взгляд) кажется бессмысленным. Возьмем BrowsableAttribute для примера; параметр boolean может показаться устаревшим, судя по имени, но возьмем этот пример:

 class SomeComponent
{
  [Browsable(true)]
  public virtual string SomeInfo{get;set;}
}

class SomeOtherComponent : SomeComponent
{
  [Browsable(false)]  // this property should not be browsable any more
  public override string SomeInfo{get;set;}
}
  

итак, чтобы ответить на ваш вопрос, вы могли бы сделать так, чтобы ваш атрибут ExternallyVisible принимал логический параметр, чтобы указать, действительно ли он виден извне, и при наследовании вы можете изменить на false — точно так же, как BrowsableAttribute .

Ответ №2:

У меня это работает.

Тестовый код:

 public static void Main()
{
    var attribute = GetAttribute(typeof (MyWebControl), "StyleString", false);
    Debug.Assert(attribute != null);

    attribute = GetAttribute(typeof(SmarterWebControl), "StyleString", false);
    Debug.Assert(attribute == null);

    attribute = GetAttribute(typeof(SmarterWebControl), "StyleString", true);
    Debug.Assert(attribute == null);
}

private static ExternallyVisibleAttribute GetAttribute(Type type, string propertyName, bool inherit)
{
    PropertyInfo property = type.GetProperties().Where(p=>p.Name.Equals(propertyName, StringComparison.OrdinalIgnoreCase)).FirstOrDefault();

    var list = property.GetCustomAttributes(typeof(ExternallyVisibleAttribute), inherit).Select(o => (ExternallyVisibleAttribute)o);

    return list.FirstOrDefault();
}
  

Ответ №3:

Я не понимаю, в чем проблема. Ваш опубликованный код выполняет ожидаемую вещь (по крайней мере, то, что, по-видимому, вы ожидаете от него) в моем тесте: то есть у StyleString свойства нет ExternallyVisible атрибута. Вот мой тестовый код:

 [AttributeUsage(AttributeTargets.Property)]
public class ExternallyVisible : Attribute
{
}

public class MyWebControl
{
    [ExternallyVisible]
    public string StyleString { get; set; }
}

public class SmarterWebControl : MyWebControl
{
    [ExternallyVisible]
    public string CssName { get; set; }

    new public string StyleString { get; set; } // Doesn't work
}

class Program
{
    static void Main()
    {
        MyWebControl myctrl = new MyWebControl();
        SmarterWebControl smartctrl = new SmarterWebControl();

        MemberInfo info = typeof(SmarterWebControl);
        PropertyInfo[] props = (typeof(SmarterWebControl)).GetProperties();
        Console.WriteLine("{0} properties", props.Length);

        foreach (var prop in props)
        {
            Console.WriteLine(prop.Name);
            foreach (var attr in prop.GetCustomAttributes(true))
            {
                Console.WriteLine("  "   attr);
            }
        }

        Console.ReadLine();
    }
}
  

В .NET 4.0 я получаю следующий вывод:

 2 properties
CssName
  sotesto.ExternallyVisible
StyleString
  

Другими словами, атрибут не применяется к StyleString свойству.

Ответ №4:

Вы могли бы унаследовать атрибут и добавить свойство, которое определяет, срабатывает ли код attrbiute или нет.

затем вы можете переопределить поведение атрибутов в унаследованном классе?

таким образом, вы бы использовали (если бы добавили параметр в конструктор)

 [ExternallyVisible(false)]

[ExternallyVisible(Enabled = false)]
  

насколько я знаю, вы использовали свойство enabled в классе atrtibute