Comment référencer un champ par réflexion
-
30-09-2019 - |
Question
Désolé pour le titre, ce n'est pas explicite.
Suite à ma question précédent, je vous abonner une méthode pour un objet d'événement récupéré dynamiquement (par réflexion). L'objet en question est un champ d'un contrôle:
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 ???
}
}
Je ne sais pas comment référencer la variable « Button1 ». J'ai essayé quelque chose comme ceci:
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 ???
}
}
Mais j'ai une exception ici:
f.GetValue(o);
System.ArgumentException a été Unhandled Message = Le champ « Button1 » défini le type « WindowsFormsApplication1.Form1 » est pas un champ de l'objet cible qui est de type « System.Windows.Forms.Button ».
La solution
C'est parce que vous essayez de créer une nouvelle instance de Button
et essayer d'obtenir la valeur de sa propriété button1
, qui évidemment n'existe pas.
Remplacer ceci:
Type t = f.FieldType;
object o = Activator.CreateInstance(t);
f.GetValue(o);
avec ceci:
object o = f.GetValue(control);
Vous pouvez utiliser une méthode comme celui-ci pour obtenir la valeur d'un champ pour tout objet:
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);
}