Как мне создать динамические свойства в C #? - PullRequest
85 голосов
/ 04 июня 2009

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

Как мне это сделать в C #?

Ответы [ 12 ]

56 голосов
/ 04 июня 2009

Вы можете использовать словарь, скажем

Dictionary<string,object> properties;

Я думаю, что в большинстве случаев, когда происходит нечто подобное, это делается так.
В любом случае вы ничего не получите от создания «реального» свойства с методами доступа set и get, поскольку оно будет создано только во время выполнения, и вы не будете использовать его в своем коде ...

Вот пример, показывающий возможную реализацию фильтрации и сортировки (без проверки ошибок):

using System;
using System.Collections.Generic;
using System.Linq;

namespace ConsoleApplication1 {

    class ObjectWithProperties {
        Dictionary<string, object> properties = new Dictionary<string,object>();

        public object this[string name] {
            get { 
                if (properties.ContainsKey(name)){
                    return properties[name];
                }
                return null;
            }
            set {
                properties[name] = value;
            }
        }

    }

    class Comparer<T> : IComparer<ObjectWithProperties> where T : IComparable {

        string m_attributeName;

        public Comparer(string attributeName){
            m_attributeName = attributeName;
        }

        public int Compare(ObjectWithProperties x, ObjectWithProperties y) {
            return ((T)x[m_attributeName]).CompareTo((T)y[m_attributeName]);
        }

    }

    class Program {

        static void Main(string[] args) {

            // create some objects and fill a list
            var obj1 = new ObjectWithProperties();
            obj1["test"] = 100;
            var obj2 = new ObjectWithProperties();
            obj2["test"] = 200;
            var obj3 = new ObjectWithProperties();
            obj3["test"] = 150;
            var objects = new List<ObjectWithProperties>(new ObjectWithProperties[]{ obj1, obj2, obj3 });

            // filtering:
            Console.WriteLine("Filtering:");
            var filtered = from obj in objects
                         where (int)obj["test"] >= 150
                         select obj;
            foreach (var obj in filtered){
                Console.WriteLine(obj["test"]);
            }

            // sorting:
            Console.WriteLine("Sorting:");
            Comparer<int> c = new Comparer<int>("test");
            objects.Sort(c);
            foreach (var obj in objects) {
                Console.WriteLine(obj["test"]);
            }
        }

    }
}
30 голосов
/ 04 июня 2009

Если вам это нужно для привязки данных, вы можете сделать это с помощью пользовательской модели дескриптора ... реализуя ICustomTypeDescriptor, TypeDescriptionProvider и / или TypeCoverter, вы можете создавать свои собственные PropertyDescriptor экземпляры во время выполнения. Это то, что элементы управления, такие как DataGridView, PropertyGrid и т. Д., Используются для отображения свойств.

Для привязки к спискам вам понадобятся ITypedList и IList; для базовой сортировки: IBindingList; для фильтрации и расширенной сортировки: IBindingListView; для полной поддержки «новой строки» (DataGridView): ICancelAddNew (фу!).

Это лот работы, хотя. DataTable (хотя я ненавижу это) - дешевый способ сделать то же самое. Если вам не требуется привязка данных, просто используйте хеш-таблицу; -p

Вот простой пример - но вы можете сделать намного больше ...

28 голосов
/ 12 августа 2011

Используйте ExpandoObject , как ViewBag в MVC 3.

12 голосов
/ 04 июня 2009

Я не уверен, что вы действительно хотите делать то, что, как вы говорите, вы хотите , но я не могу объяснить, почему!

Вы не можете добавить свойства в класс после того, как он был JITed.

Самое близкое, что вы могли бы получить, это динамически создать подтип с помощью Reflection.Emit и скопировать существующие поля, но вам придется обновить все ссылки на объект самостоятельно.

Вы также не сможете получить доступ к этим свойствам во время компиляции.

Что-то вроде:

public class Dynamic
{
    public Dynamic Add<T>(string key, T value)
    {
        AssemblyBuilder assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(new AssemblyName("DynamicAssembly"), AssemblyBuilderAccess.Run);
        ModuleBuilder moduleBuilder = assemblyBuilder.DefineDynamicModule("Dynamic.dll");
        TypeBuilder typeBuilder = moduleBuilder.DefineType(Guid.NewGuid().ToString());
        typeBuilder.SetParent(this.GetType());
        PropertyBuilder propertyBuilder = typeBuilder.DefineProperty(key, PropertyAttributes.None, typeof(T), Type.EmptyTypes);

        MethodBuilder getMethodBuilder = typeBuilder.DefineMethod("get_" + key, MethodAttributes.Public, CallingConventions.HasThis, typeof(T), Type.EmptyTypes);
        ILGenerator getter = getMethodBuilder.GetILGenerator();
        getter.Emit(OpCodes.Ldarg_0);
        getter.Emit(OpCodes.Ldstr, key);
        getter.Emit(OpCodes.Callvirt, typeof(Dynamic).GetMethod("Get", BindingFlags.Instance | BindingFlags.NonPublic).MakeGenericMethod(typeof(T)));
        getter.Emit(OpCodes.Ret);
        propertyBuilder.SetGetMethod(getMethodBuilder);

        Type type = typeBuilder.CreateType();

        Dynamic child = (Dynamic)Activator.CreateInstance(type);
        child.dictionary = this.dictionary;
        dictionary.Add(key, value);
        return child;
    }

    protected T Get<T>(string key)
    {
        return (T)dictionary[key];
    }

    private Dictionary<string, object> dictionary = new Dictionary<string,object>();
}

У меня не установлено VS на этом компьютере, поэтому дайте мне знать, если есть какие-либо серьезные ошибки (ну ... кроме проблем с производительностью, но я не написал спецификацию!)

Теперь вы можете использовать его:

Dynamic d = new Dynamic();
d = d.Add("MyProperty", 42);
Console.WriteLine(d.GetType().GetProperty("MyProperty").GetValue(d, null));

Вы также можете использовать его как обычное свойство в языке, поддерживающем позднюю привязку (например, VB.NET)

12 голосов
/ 04 июня 2009

Создайте Hashtable под названием «Свойства» и добавьте в него свои свойства.

3 голосов
/ 06 января 2011

Я сделал именно это с помощью интерфейса ICustomTypeDescriptor и словаря.

Реализация ICustomTypeDescriptor для динамических свойств:

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

Это может быть достигнуто за счет того, что каждая строка данных представляет собой словарь, ключом которого является имя свойства, а значением - строка или класс, который может хранить значение свойства для указанной строки. Конечно, наличие списка объектов словаря не может быть привязано к сетке. Вот здесь и вступает ICustomTypeDescriptor.

Путем создания класса-оболочки для словаря и его привязки к интерфейсу ICustomTypeDescriptor поведение для возврата свойств объекта может быть переопределено.

Взгляните на реализацию класса данных строки ниже:

/// <summary>
/// Class to manage test result row data functions
/// </summary>
public class TestResultRowWrapper : Dictionary<string, TestResultValue>, ICustomTypeDescriptor
{
    //- METHODS -----------------------------------------------------------------------------------------------------------------

    #region Methods

    /// <summary>
    /// Gets the Attributes for the object
    /// </summary>
    AttributeCollection ICustomTypeDescriptor.GetAttributes()
    {
        return new AttributeCollection(null);
    }

    /// <summary>
    /// Gets the Class name
    /// </summary>
    string ICustomTypeDescriptor.GetClassName()
    {
        return null;
    }

    /// <summary>
    /// Gets the component Name
    /// </summary>
    string ICustomTypeDescriptor.GetComponentName()
    {
        return null;
    }

    /// <summary>
    /// Gets the Type Converter
    /// </summary>
    TypeConverter ICustomTypeDescriptor.GetConverter()
    {
        return null;
    }

    /// <summary>
    /// Gets the Default Event
    /// </summary>
    /// <returns></returns>
    EventDescriptor ICustomTypeDescriptor.GetDefaultEvent()
    {
        return null;
    }

    /// <summary>
    /// Gets the Default Property
    /// </summary>
    PropertyDescriptor ICustomTypeDescriptor.GetDefaultProperty()
    {
        return null;
    }

    /// <summary>
    /// Gets the Editor
    /// </summary>
    object ICustomTypeDescriptor.GetEditor(Type editorBaseType)
    {
        return null;
    }

    /// <summary>
    /// Gets the Events
    /// </summary>
    EventDescriptorCollection ICustomTypeDescriptor.GetEvents(Attribute[] attributes)
    {
        return new EventDescriptorCollection(null);
    }

    /// <summary>
    /// Gets the events
    /// </summary>
    EventDescriptorCollection ICustomTypeDescriptor.GetEvents()
    {
        return new EventDescriptorCollection(null);
    }

    /// <summary>
    /// Gets the properties
    /// </summary>
    PropertyDescriptorCollection ICustomTypeDescriptor.GetProperties(Attribute[] attributes)
    {
        List<propertydescriptor> properties = new List<propertydescriptor>();

        //Add property descriptors for each entry in the dictionary
        foreach (string key in this.Keys)
        {
            properties.Add(new TestResultPropertyDescriptor(key));
        }

        //Get properties also belonging to this class also
        PropertyDescriptorCollection pdc = TypeDescriptor.GetProperties(this.GetType(), attributes);

        foreach (PropertyDescriptor oPropertyDescriptor in pdc)
        {
            properties.Add(oPropertyDescriptor);
        }

        return new PropertyDescriptorCollection(properties.ToArray());
    }

    /// <summary>
    /// gets the Properties
    /// </summary>
    PropertyDescriptorCollection ICustomTypeDescriptor.GetProperties()
    {
        return ((ICustomTypeDescriptor)this).GetProperties(null);
    }

    /// <summary>
    /// Gets the property owner
    /// </summary>
    object ICustomTypeDescriptor.GetPropertyOwner(PropertyDescriptor pd)
    {
        return this;
    }

    #endregion Methods

    //---------------------------------------------------------------------------------------------------------------------------
}

Примечание. В методе GetProperties я могу кэшировать PropertyDescriptors, когда-то считываемые для повышения производительности, но, поскольку я добавляю и удаляю столбцы во время выполнения, я всегда хочу их перестроить

В методе GetProperties вы также заметите, что дескрипторы свойств, добавленные для словарных статей, имеют тип TestResultPropertyDescriptor. Это пользовательский класс дескриптора свойства, который управляет настройкой и получением свойств. Посмотрите на реализацию ниже:

/// <summary>
/// Property Descriptor for Test Result Row Wrapper
/// </summary>
public class TestResultPropertyDescriptor : PropertyDescriptor
{
    //- PROPERTIES --------------------------------------------------------------------------------------------------------------

    #region Properties

    /// <summary>
    /// Component Type
    /// </summary>
    public override Type ComponentType
    {
        get { return typeof(Dictionary<string, TestResultValue>); }
    }

    /// <summary>
    /// Gets whether its read only
    /// </summary>
    public override bool IsReadOnly
    {
        get { return false; }
    }

    /// <summary>
    /// Gets the Property Type
    /// </summary>
    public override Type PropertyType
    {
        get { return typeof(string); }
    }

    #endregion Properties

    //- CONSTRUCTOR -------------------------------------------------------------------------------------------------------------

    #region Constructor

    /// <summary>
    /// Constructor
    /// </summary>
    public TestResultPropertyDescriptor(string key)
        : base(key, null)
    {

    }

    #endregion Constructor

    //- METHODS -----------------------------------------------------------------------------------------------------------------

    #region Methods

    /// <summary>
    /// Can Reset Value
    /// </summary>
    public override bool CanResetValue(object component)
    {
        return true;
    }

    /// <summary>
    /// Gets the Value
    /// </summary>
    public override object GetValue(object component)
    {
          return ((Dictionary<string, TestResultValue>)component)[base.Name].Value;
    }

    /// <summary>
    /// Resets the Value
    /// </summary>
    public override void ResetValue(object component)
    {
        ((Dictionary<string, TestResultValue>)component)[base.Name].Value = string.Empty;
    }

    /// <summary>
    /// Sets the value
    /// </summary>
    public override void SetValue(object component, object value)
    {
        ((Dictionary<string, TestResultValue>)component)[base.Name].Value = value.ToString();
    }

    /// <summary>
    /// Gets whether the value should be serialized
    /// </summary>
    public override bool ShouldSerializeValue(object component)
    {
        return false;
    }

    #endregion Methods

    //---------------------------------------------------------------------------------------------------------------------------
}

Основные свойства этого класса - это GetValue и SetValue. Здесь вы видите компонент, который преобразуется в словарь, а значение ключа внутри него устанавливается или восстанавливается. Важно, чтобы словарь в этом классе был того же типа, что и класс-оболочка Row, иначе приведение не удастся. Когда дескриптор создан, ключ (имя свойства) передается и используется для запроса словаря для получения правильного значения.

Взято из моего блога по адресу:

ICustomTypeDescriptor Реализация динамических свойств

1 голос
/ 08 июля 2009

В качестве замены для некоторого кода orsogufo, поскольку я недавно сам использовал словарь для этой же проблемы, вот мой оператор []:

public string this[string key]
{
    get { return properties.ContainsKey(key) ? properties[key] : null; }

    set
    {
        if (properties.ContainsKey(key))
        {
            properties[key] = value;
        }
        else
        {
            properties.Add(key, value);
        }
    }
}

В этой реализации установщик добавит новые пары ключ-значение при использовании []=, если они еще не существуют в словаре.

Кроме того, для меня properties - это IDictionary, а в конструкторах я инициализирую его new SortedDictionary<string, string>().

1 голос
/ 04 июня 2009

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

Еще одна полезная вещь, которую стоит посмотреть - CSLA.Net . Код находится в свободном доступе и использует некоторые из принципов \ шаблонов, которые, по вашему мнению, вам нужны.

Также, если вы смотрите на сортировку и фильтрацию, я предполагаю, что вы будете использовать какую-то сетку. Полезный интерфейс для реализации - ICustomTypeDescriptor, он позволяет эффективно переопределять то, что происходит, когда ваш объект отражается, поэтому вы можете указать отражатель на собственную внутреннюю хеш-таблицу вашего объекта.

1 голос
/ 04 июня 2009

Я не уверен, каковы ваши причины, и даже если бы вы могли как-то это осуществить с помощью Reflection Emit (я не уверен, что вы можете), это не звучит как хорошая идея. Вероятно, лучшая идея - иметь некоторый словарь, и вы можете обернуть доступ к словарю через методы в вашем классе. Таким образом, вы можете сохранить данные из базы данных в этом словаре, а затем извлечь их, используя эти методы.

0 голосов
/ 15 апреля 2011

Если это для привязки, то вы можете ссылаться на индексаторы из XAML

Text="{Binding [FullName]}"

Здесь это ссылка на индексатор классов с ключом "FullName"

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...