Как получить список свойств класса?


Ответы:


800

Отражение; например:

obj.GetType().GetProperties();

для типа:

typeof(Foo).GetProperties();

например:

class Foo {
    public int A {get;set;}
    public string B {get;set;}
}
...
Foo foo = new Foo {A = 1, B = "abc"};
foreach(var prop in foo.GetType().GetProperties()) {
    Console.WriteLine("{0}={1}", prop.Name, prop.GetValue(foo, null));
}

Следующие отзывы ...

  • Чтобы получить значение статических свойств, передайте nullв качестве первого аргументаGetValue
  • Чтобы просмотреть непубличные свойства, используйте (например) GetProperties(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance)(который возвращает все публичные / частные свойства экземпляра).

13
Для полноты, есть также ComponentModel, предоставляемый TypeDescriptor.GetProperties (...) - который позволяет динамические свойства времени выполнения (отражение фиксируется во время компиляции).
Марк Гравелл

5
Предложение: Разверните ответ, чтобы охватить защищенные / частные / статические / унаследованные свойства.
Ричард

1
Выражение foreach, которое вы отображаете, работает даже внутри класса, свойства которого вы хотите получить :)
halfpastfour.am

Мне было непонятно, как указаны дополнительные комментарии, но использование всех 3-х флагов также дает вам internalсвойства. Может быть, я единственный, кто зациклен на private/ non-publicсинтаксис?
Бричинс

1
@Tadej, на какие рамки вы ориентируетесь? если вы используете ядро ​​.NET, вам нужно убедиться, что у вас есть using System.Reflectionдиректива и System.Reflection.TypeExtensionsпакет, на который есть ссылка - это обеспечивает недостающую поверхность API с помощью методов расширения
Марк Грэвелл

92

Вы можете использовать Reflection, чтобы сделать это: (из моей библиотеки - это получает имена и значения)

public static Dictionary<string, object> DictionaryFromType(object atype)
{
    if (atype == null) return new Dictionary<string, object>();
    Type t = atype.GetType();
    PropertyInfo[] props = t.GetProperties();
    Dictionary<string, object> dict = new Dictionary<string, object>();
    foreach (PropertyInfo prp in props)
    {
        object value = prp.GetValue(atype, new object[]{});
        dict.Add(prp.Name, value);
    }
    return dict;
}

Эта вещь не будет работать для свойств с индексом - для этого (это становится громоздким):

public static Dictionary<string, object> DictionaryFromType(object atype, 
     Dictionary<string, object[]> indexers)
{
    /* replace GetValue() call above with: */
    object value = prp.GetValue(atype, ((indexers.ContainsKey(prp.Name)?indexers[prp.Name]:new string[]{});
}

Кроме того, чтобы получить только открытые свойства: ( см. MSDN в перечислении BindingFlags )

/* replace */
PropertyInfo[] props = t.GetProperties();
/* with */
PropertyInfo[] props = t.GetProperties(BindingFlags.Public)

Это работает и для анонимных типов!
Чтобы просто получить имена:

public static string[] PropertiesFromType(object atype)
{
    if (atype == null) return new string[] {};
    Type t = atype.GetType();
    PropertyInfo[] props = t.GetProperties();
    List<string> propNames = new List<string>();
    foreach (PropertyInfo prp in props)
    {
        propNames.Add(prp.Name);
    }
    return propNames.ToArray();
}

И это примерно то же самое только для значений, или вы можете использовать:

GetDictionaryFromType().Keys
// or
GetDictionaryFromType().Values

Но это немного медленнее, я думаю.


... но atype.GetProperty (prp.Name) собирается вернуть prp?
Марк Гравелл

5
Что касается бита общедоступных свойств, в соответствии со связанной статьей MSDN: «Примечание. Вы должны указать Instance или Static вместе с Public или NonPublic, иначе члены не будут возвращены». Таким образом, пример кода должен быть: t.GetProperties(BindingFlags.Instance | BindingFlags.Public)илиt.GetProperties(BindingFlags.Static | BindingFlags.Public)
Карл Шарман

я не искал код, я искал объяснение рефлексии и вау, так высоко ценится! Сделайте это универсальным, и вы можете просто сказать, что ваша программа обладает сверхспособностями;)
Jaquarh

37
public List<string> GetPropertiesNameOfClass(object pObject)
{
    List<string> propertyList = new List<string>();
    if (pObject != null)
    {
        foreach (var prop in pObject.GetType().GetProperties())
        {
            propertyList.Add(prop.Name);
        }
    }
    return propertyList;
}

Эта функция предназначена для получения списка свойств класса.


7
Вы можете изменить это, чтобы использовать yield return. Это не имеет большого значения, но это лучший способ сделать это.
Мэтью Хауген

1
Мне это нравится, потому что это (почти) единственный ответ, который не включает в себя слово отражение .

9
Но это все еще использует отражение все же.
GGG

2
я полагаю, это гораздо лучше pObject.GetType (). GetProperties (). Select (p => p.Name)
разочарован

23

Вы можете использовать System.Reflectionпространство имен с Type.GetProperties()методом:

PropertyInfo[] propertyInfos;
propertyInfos = typeof(MyClass).GetProperties(BindingFlags.Public|BindingFlags.Static);

23

Основываясь на ответе @ MarcGravell, вот версия, которая работает в Unity C #.

ObjectsClass foo = this;
foreach(var prop in foo.GetType().GetProperties()) {
    Debug.Log("{0}={1}, " + prop.Name + ", " + prop.GetValue(foo, null));
}

8

Это мое решение

public class MyObject
{
    public string value1 { get; set; }
    public string value2 { get; set; }

    public PropertyInfo[] GetProperties()
    {
        try
        {
            return this.GetType().GetProperties();
        }
        catch (Exception ex)
        {

            throw ex;
        }
    }

    public PropertyInfo GetByParameterName(string ParameterName)
    {
        try
        {
            return this.GetType().GetProperties().FirstOrDefault(x => x.Name == ParameterName);
        }
        catch (Exception ex)
        {

            throw ex;
        }
    }

    public static MyObject SetValue(MyObject obj, string parameterName,object parameterValue)
    {
        try
        {
            obj.GetType().GetProperties().FirstOrDefault(x => x.Name == parameterName).SetValue(obj, parameterValue);
            return obj;
        }
        catch (Exception ex)
        {
            throw ex;
        }
    }
}

6

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

Type typeOfMyObject = myObject.GetType();
PropertyInfo[] properties =typeOfMyObject.GetProperties();

3

Я также сталкиваюсь с такого рода требованиями.

Из этой дискуссии я получил еще одну идею,

Obj.GetType().GetProperties()[0].Name

Это также показывает имя свойства.

Obj.GetType().GetProperties().Count();

это показывает количество свойств.

Спасибо всем. Это хорошая дискуссия.


3

Вот улучшенный ответ @lucasjones. Я включил улучшения, упомянутые в разделе комментариев после его ответа. Я надеюсь, что кто-то найдет это полезным.

public static string[] GetTypePropertyNames(object classObject,  BindingFlags bindingFlags)
{
    if (classObject == null)
    {
        throw new ArgumentNullException(nameof(classObject));
    }

        var type = classObject.GetType();
        var propertyInfos = type.GetProperties(bindingFlags);

        return propertyInfos.Select(propertyInfo => propertyInfo.Name).ToArray();
 }
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.