Iterate through properties and values of an object returned via a linq query on a domain model

后端 未结 3 2008
旧巷少年郎
旧巷少年郎 2021-02-01 01:26

I have a custom entity in a relational database that I have mapped to the CLR via a domain model. So by using the following statement, I can pull in an entity from my database i

3条回答
  •  无人及你
    2021-02-01 02:09

    I assume that you're trying to define a general-purpose way to "dump" an object without knowing anything about its structure. If so, then you are going about things the correct way. You use reflection (GetType() and the associated Type class methods) to inspect the object and return its information.

    The reason GetFields() didn't return anything is that you likely did not supply the right binding flags. In particular, if you call the overload that doesn't take any parameters, you only get back public fields; if you want private fields you need to ask for them specifically.

    In your case, GetFields(BindingFlags.NonPublic) would give you back the _new_systemGauges and _new_systemAlarm fields, while GetProperties() would give you back the New_systemAlarm and New_systemAlarm properties.

    The other key element you missed is that the data you are getting back is the type metadata; it defines the structure of the class, and not any particular instance. If you want to know what the value of a property for a specific instance is, you need to ask for that:

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

    One you have one of the PropertyInfo elements from the type's metadata, you can ask for that property value on any instance of that type. It doesn't have to be the same instance that you originally used. For example:

    var objs = somelist.Where(x => x.Id == 1);
    foreach (var prop in objs.First().GetType().GetProperties())
    {
      int x = 0;
      foreach (var obj in objs)
      {        
        if (prop.PropertyType.Name.Equals("Int32"))
        {
          int val = (int)prop.GetValue(obj, null);
          Console.WriteLine("Obj #{0}: {1} = 0x{2:x8}", x++, prop.Name, val);
        }
        else if (prop.PropertyType.Name.Equals("Decimal"))
        {
          int val = (decimal)prop.GetValue(obj, null);
          Console.WriteLine("Obj #{0}: {1} = {2:c2}", x++, prop.Name, val);
        }
        else
        {
          Console.WriteLine("Obj #{0}: {1} = '{2}'", x++, prop.Name, prop.GetValue(obj, null));
        }
      }
    }
    

    Technically you should check the result of GetIndexParameters to see if a property is indexed or not; the null parameter to GetValue is actually an array of index values.

    To convert the value you get back you can either use typecasts, or if you want to be a bit more flexible, use the Convert class's methods. The difference is, for example, if you have a short property, GetValue() will return a boxed short, which you cannot then typecast as an int; you have to unbox it to a short first. Using Convert.ToInt32() will perform all of the needed steps to get an int value out of any property that is convertible to an integer.

    Converting between reference types is easier since you can just use is and as for that; those work just like you'd expect with "reflected" property values.

提交回复
热议问题