C # - получить все свойства родового типа путем отражения [duplicate]

Для тех, кто не может полагаться на Chronometer , я сделал класс утилиты из одного из предложений:

public class TimerTextHelper implements Runnable {
   private final Handler handler = new Handler();
   private final TextView textView;
   private volatile long startTime;
   private volatile long elapsedTime;

   public TimerTextHelper(TextView textView) {
       this.textView = textView;
   }

   @Override
   public void run() {
       long millis = System.currentTimeMillis() - startTime;
       int seconds = (int) (millis / 1000);
       int minutes = seconds / 60;
       seconds = seconds % 60;

       textView.setText(String.format("%d:%02d", minutes, seconds));

       if (elapsedTime == -1) {
           handler.postDelayed(this, 500);
       }
   }

   public void start() {
       this.startTime = System.currentTimeMillis();
       this.elapsedTime = -1;
       handler.post(this);
   }

   public void stop() {
       this.elapsedTime = System.currentTimeMillis() - startTime;
       handler.removeCallbacks(this);
   }

   public long getElapsedTime() {
       return elapsedTime;
   }

}

для использования.. просто делать:

 TimerTextHelper timerTextHelper = new TimerTextHelper(textView);
 timerTextHelper.start();

.....

 timerTextHelper.stop();
 long elapsedTime = timerTextHelper.getElapsedTime();

485
задан nawfal 16 June 2013 в 09:37
поделиться

9 ответов

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

Type typeOfMyObject = myObject.GetType();
PropertyInfo[] properties =typeOfMyObject.GetProperties();
6
ответ дан Daan 21 August 2018 в 08:25
поделиться

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

ObjectsClass foo = this;
foreach(var prop in foo.GetType().GetProperties()) {
    Debug.Log("{0}={1}, " + prop.Name + ", " + prop.GetValue(foo, null));
}
14
ответ дан Jacksonkr 21 August 2018 в 08:25
поделиться

Вы можете использовать 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 enum )

/* 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

Но это немного медленнее, я бы представьте себе.

61
ответ дан JHobern 21 August 2018 в 08:25
поделиться
  • 1
    ... но atype.GetProperty (prp.Name) собирается вернуть prp? – Marc Gravell♦ 10 April 2009 в 10:37
  • 2
    О, ты прав! D'о. – Lucas Jones 10 April 2009 в 10:43
  • 3
    Я исправил это сейчас. – Lucas Jones 10 April 2009 в 10:57
  • 4
    Что касается бита публичных свойств, в соответствии со связанной статьей MSDN: & quot; Примечание. Необходимо указать экземпляр или Static вместе с Public или NonPublic, или никакие члены не будут возвращены. & Quot ;. Таким образом, пример кода должен быть: t.GetProperties(BindingFlags.Instance | BindingFlags.Public) или t.GetProperties(BindingFlags.Static | BindingFlags.Public) – Carl 6 May 2015 в 15:01
  • 5
    не искал код, я искал объяснения для размышлений и ничего себе, так высоко оценил! Сделайте это родовым, и вы могли бы просто сказать, что ваша программа обладает суперспособностями;) – KDOT 18 March 2016 в 17:47

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

PropertyInfo[] propertyInfos;
propertyInfos = typeof(MyClass).GetProperties(BindingFlags.Public|BindingFlags.Static);
20
ответ дан Jon Limjap 21 August 2018 в 08:25
поделиться

Здесь улучшен ответ @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();
 }
3
ответ дан Kjartan 21 August 2018 в 08:25
поделиться

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

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;
        }
    }
}
31
ответ дан Pang 21 August 2018 в 08:25
поделиться
  • 1
    Вы можете изменить это, чтобы использовать yield return. Это неважно, но это лучший способ сделать это. – Matthew Haugen 24 July 2014 в 04:42
  • 2
    Мне это нравится, потому что это (почти) единственный ответ, который не включает слово reflection . – user 28 July 2015 в 15:41
  • 3
    Но все же использует отражение. – GGG 12 April 2016 в 21:51
  • 4
    я думаю, это намного лучше pObject.GetType (). GetProperties (). Выберите (p = & gt; p.Name) – Disappointed 9 June 2016 в 16:18

Отражение; для экземпляра:

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) (который возвращает все свойства public / private экземпляра).
631
ответ дан Robert Harvey 21 August 2018 в 08:25
поделиться
  • 1
    Для полноты есть также ComponentModel, открытый TypeDescriptor.GetProperties (...) - который позволяет динамические свойства времени выполнения (отражение фиксировано во время компиляции). – Marc Gravell♦ 10 April 2009 в 10:38
  • 2
    Предложение. Разверните ответ, чтобы защитить защищенные / частные / статические / унаследованные свойства. – Richard 10 April 2009 в 10:39
  • 3
    Заявление foreach, которое вы показываете, даже работает из класса, в котором вы хотите получить свойства :) – halfpastfour.am 24 January 2012 в 11:27
  • 4
    Мне было непонятно, как указываются дополнительные комментарии, но использование всех трех флагов также дает вам свойства internal. Может быть, я единственный, кто повесил трубку на синтаксисе private / non-public? – brichins 9 August 2013 в 00:39
  • 5
    @Tadej, какие рамки вы нацеливаете? если вы используете ядро ​​.NET, вам необходимо убедиться, что вы указали директиву using System.Reflection и System.Reflection.TypeExtensions - это обеспечивает отсутствие поверхности API через методы расширения – Marc Gravell♦ 17 May 2017 в 09:18

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

Из этого обсуждения я получил еще одну идею,

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

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

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

, показывающий количество свойств.

Спасибо всем. Это приятное обсуждение.

3
ответ дан Singaravelan 21 August 2018 в 08:25
поделиться
33
ответ дан Pang 1 November 2018 в 03:12
поделиться
Другие вопросы по тегам:

Похожие вопросы: