Come fare riferimento a un campo dalla riflessione
-
30-09-2019 - |
Domanda
Ci scusiamo per il titolo, non è esplicito.
Con riferimento alla mia precedente domanda , ho utile per iscriversi un metodo per un oggetto evento recuperati in modo dinamico (attraverso la riflessione). L'oggetto in questione è un campo di un controllo:
public void SubscribeEvents(Control control)
{
Type controlType = control.GetType();
FieldInfo[] fields = controlType.GetFields(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
MethodInfo method = typeof(Trace).GetMethod("WriteTrace");
// "button1" hardcoded for the sample
FieldInfo f = controlType.GetField("button1", BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
// "Click" hardcoded for the sample
EventInfo eInfo = f.FieldType.GetEvent("Click");
if (eInfo != null)
{
EventHandler dummyDelegate = (s, e) => WriteTrace(s, e, eInfo.Name);
Delegate realDelegate = Delegate.CreateDelegate(eInfo.EventHandlerType, dummyDelegate.Target, dummyDelegate.Method);
eInfo.AddEventHandler(?????, realDelegate); // How can I reference the variable button1 ???
}
}
Non so come fare riferimento alla variabile 'button1'. Ho provato qualcosa di simile:
public void SubscribeEvents(Control control)
{
Type controlType = control.GetType();
FieldInfo[] fields = controlType.GetFields(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
MethodInfo method = typeof(Trace).GetMethod("WriteTrace");
// "button1" hardcoded for the sample
FieldInfo f = controlType.GetField("button1", BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
// "Click" hardcoded for the sample
EventInfo eInfo = f.FieldType.GetEvent("Click");
Type t = f.FieldType;
object o = Activator.CreateInstance(t);
f.GetValue(o);
if (eInfo != null)
{
EventHandler dummyDelegate = (s, e) => WriteTrace(s, e, eInfo.Name);
Delegate realDelegate = Delegate.CreateDelegate(eInfo.EventHandlerType, dummyDelegate.Target, dummyDelegate.Method);
eInfo.AddEventHandler(o, realDelegate); // Why can I refer to the variable button1 ???
}
}
Ma ho un'eccezione qui:
f.GetValue(o);
System.ArgumentException è stata gestita Messaggio = Campo 'button1' definita sul tipo 'WindowsFormsApplication1.Form1' non è un campo sull'oggetto di destinazione che è di tipo 'System.Windows.Forms.Button'.
Soluzione
Questo perché si sta cercando di creare una nuova istanza di Button
e cercando di ottenere il valore della sua proprietà button1
, che ovviamente non esiste.
Sostituire questa:
Type t = f.FieldType;
object o = Activator.CreateInstance(t);
f.GetValue(o);
con questo:
object o = f.GetValue(control);
È possibile utilizzare un metodo come questo per ottenere il valore di un campo per qualsiasi oggetto:
public static T GetFieldValue<T>(object obj, string fieldName)
{
if (obj == null)
throw new ArgumentNullException("obj");
var field = obj.GetType().GetField(fieldName, BindingFlags.Public |
BindingFlags.NonPublic |
BindingFlags.Instance);
if (field == null)
throw new ArgumentException("fieldName", "No such field was found.");
if (!typeof(T).IsAssignableFrom(field.FieldType))
throw new InvalidOperationException("Field type and requested type are not compatible.");
return (T)field.GetValue(obj);
}