Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How do I get the value of MemberInfo?

Tags:

c#

reflection

How do I get the value of a MemberInfo object? .Name returns the name of the variable, but I need the value.

I think you can do this with FieldInfo but I don't have a snippet, if you know how to do this can you provide a snippet??

Thanks!

like image 321
Keith Fitzgerald Avatar asked Oct 26 '08 20:10

Keith Fitzgerald


3 Answers

Although I generally agree with Marc's point about not reflecting fields, there are times when it is needed. If you want to reflect a member and you don't care whether it is a field or a property, you can use this extension method to get the value (if you want the type instead of the value, see nawful's answer to this question):

    public static object GetValue(this MemberInfo memberInfo, object forObject)
    {
        switch (memberInfo.MemberType)
        {
            case MemberTypes.Field:
                return ((FieldInfo)memberInfo).GetValue(forObject);
            case MemberTypes.Property:
                return ((PropertyInfo)memberInfo).GetValue(forObject);
            default:
                throw new NotImplementedException();
        }
    } 
like image 54
Eric Hewitt Avatar answered Nov 15 '22 08:11

Eric Hewitt


Here's an example for fields, using FieldInfo.GetValue:

using System;
using System.Reflection;

public class Test
{
    // public just for the sake of a short example.
    public int x;

    static void Main()
    {
        FieldInfo field = typeof(Test).GetField("x");
        Test t = new Test();
        t.x = 10;

        Console.WriteLine(field.GetValue(t));
    }
}

Similar code will work for properties using PropertyInfo.GetValue() - although there you also need to pass the values for any parameters to the properties. (There won't be any for "normal" C# properties, but C# indexers count as properties too as far as the framework is concerned.) For methods, you'll need to call Invoke if you want to call the method and use the return value.

like image 43
Jon Skeet Avatar answered Nov 15 '22 07:11

Jon Skeet


Jon's answer is ideal - just one observation: as part of general design, I would:

  1. generally avoid reflecting against non-public members
  2. avoid having public fields (almost always)

The upshot of these two is that generally you only need to reflect against public properties (you shouldn't be calling methods unless you know what they do; property getters are expected to be idempotent [lazy loading aside]). So for a PropertyInfo this is just prop.GetValue(obj, null);.

Actually, I'm a big fan of System.ComponentModel, so I would be tempted to use:

    foreach(PropertyDescriptor prop in TypeDescriptor.GetProperties(obj))
    {
        Console.WriteLine("{0}={1}", prop.Name, prop.GetValue(obj));
    }

or for a specific property:

    PropertyDescriptor prop = TypeDescriptor.GetProperties(obj)["SomeProperty"];
    Console.WriteLine("{0}={1}", prop.Name, prop.GetValue(obj));

One advantage of System.ComponentModel is that it will work with abstracted data models, such as how a DataView exposes columns as virtual properties; there are other tricks too (like performance tricks).

like image 13
Marc Gravell Avatar answered Nov 15 '22 09:11

Marc Gravell