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 into memory via a LINQ query on the domain model, like so;
var inspection = (from i in dbContext.New_testinspectionExtensionBases
where i.New_testinspectionId == currentInspection
select i).First();
There are properties/fields on this entity that I need access to, I need to be able to determine the property/field name as well as it’s value. I want to loop through these items in memory, and write out their names and values to the console.
I tried using this approach, but couldn’t figure out how to correct the syntax (Nor am I sure that GetProperties is the correct method to use, GetFields wasn’t returning anything for some reason so I assumed this was the way to go) but it doesn’t really matter since all i need is read access to the value;
var inspectionReportFields = inspection.GetType().GetProperties();
// I called this inspectionReportfields because the entity properties correspond to
// form/report fields I'm generating from this data.
foreach (var reportField in inspectionReportFields)
{
var value = reportField.GetValue();
Console.WriteLine(reportField.Name);
Console.WriteLine(value);
}
Is there an easier way to get the property/field value when utilizing a domain model like EF or openaccess? If not, am I going about it the right way? And lastly, if so, how do I fix the syntax in the value variable declaration?
Here are some sample fields/properties from the code generated by the domain model, for reference;
private int? _new_systemGauges;
public virtual int? New_systemGauges
{
get
{
return this._new_systemGauges;
}
set
{
this._new_systemGauges = value;
}
}
private int? _new_systemAlarm ;
public virtual int? New_systemAlarm
{
get
{
return this._new_systemAlarm;
}
set
{
this._new_systemAlarm = value;
}
}
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 associatedTypeclass 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 backpublicfields; if you want private fields you need to ask for them specifically.In your case,
GetFields(BindingFlags.NonPublic)would give you back the_new_systemGaugesand_new_systemAlarmfields, while GetProperties() would give you back theNew_systemAlarmandNew_systemAlarmproperties.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:One you have one of the
PropertyInfoelements 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:Technically you should check the result of
GetIndexParametersto see if a property is indexed or not; thenullparameter toGetValueis 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
shortproperty,GetValue()will return a boxed short, which you cannot then typecast as anint; you have to unbox it to ashortfirst. UsingConvert.ToInt32()will perform all of the needed steps to get anintvalue out of any property that is convertible to an integer.Converting between reference types is easier since you can just use
isandasfor that; those work just like you’d expect with “reflected” property values.