Diskuze: Generické metody

C# .NET .NET (C# a Visual Basic) Generické metody American English version English version

Avatar
jan.pichl
Člen
Avatar
jan.pichl:

Ahoj mám objekt

 public class Presentation_Container : INotifyPropertyChanged
    {
 private EPresentationType _PresentationType;
        public EPresentationType PresentationType
        {
            get { return _PresentationType; }
            set { _PresentationType = value; NotifyPropertyChanged("PresentationType"); }
        }

private int _MinTimeToChange;
        public int MinTimeToChange
        {
            get { return _MinTimeToChange; }
            set { _MinTimeToChange = value; NotifyPropertyChanged("MinTimeToChange"); }
        }

        }

a jinde chci měnit jeho vlastnosti Generickou metodou

public static void ChangeElement<T>( ref T item, ref T newValue) // where T: Presentation_Container
       {
           item = newValue;
       }

kterou volám

var x = (sender as Button).DataContext as Presentation_Container;

            if (x != null)
            {
  int newValue = x.MinTimeToChange+5;
ChangeElement<int>(ref  x.MinTimeToChange,   newValue);
}

A dostanu hlášení "Vlastnost nebo indexer nejde předat jako parametr ref nebo out".

Chtěl bych dosáhnout toho že jednou metodou budu měnit parametry u objektu, tak abych se vyhnul konstrukci. Že předám název parametru jako string a pak to přes case vyhodnotím a udělám přiřazení.
Tuším že generická metoda by mohla být ta spávná cesta jak toho dosáhnout. Neporadil by někdo správné použítí? předem děkuji.

Editováno 3. října 9:50
 
Odpovědět 3. října 9:49
Avatar
Marian Benčat
Redaktor
Avatar
Marian Benčat:

Eh.. Pockej.. jestli to spravne chapu..

misto

a = 5;

chces delat

Change<int>(ref a, 5) ;?

:-O

JInak k tvoji otázce... properta je syntactic sugar k metodě.. ref a outem můžeš předat proměnnou, ne metodu. Proto ti to nejde.. ono to moc nedává smysl, když si uvědomíš, že je to metoda :-)

 
Nahoru Odpovědět 3. října 11:14
Avatar
jan.pichl
Člen
Avatar
Odpovídá na Marian Benčat
jan.pichl:

Pravda to mě nedocvaklo. A nabízí se tedy otázka jak předám konkrétní (propertu)metodu odkazem a ne hodnotou.

Jde mi totiž o to že nechci měnit hodnotu přímo a = 5; ale přez univerzální metodu která krom toho že provede přířazení vyvolá i nějakou akci. Například uložení. Vým že tu funkcionalitu mohu doplnit přímo do těla property. to ovšem musím doplnit do každé property defakto to samé. Takhle bych to měl zapouzdřené na jednom místě.

 
Nahoru Odpovědět 3. října 11:37
Avatar
Marian Benčat
Redaktor
Avatar
Odpovídá na jan.pichl
Marian Benčat:

Tak na to bude nejlepší asi Expression:

public  void SetPropertyValue<T>(T target, Expression<Func<T, object>> member, object value)
  {
      var selector= member.Body as MemberExpression;
      if (selector!= null)
      {
          var property = selector.Member as PropertyInfo;
          if (property != null)
          {
              property.SetValue(target, value, null);
          }
      }
  }

jinak tydle veci co ty potrebujes se casto delaji nejakym method interception..... Doporučuji se kouknout třeba na PostSharp a na Examble: OnPropertyChanged .. ktery je defakto eliminace prave boilerplatu u INotify..objectu..

Na otm zaklade jsi shcopny udelat co potrebujes a je to mnohem efektivnejsi, nez je expression a reflexe.

Akceptované řešení
+20 Zkušeností
+1 bodů
Řešení problému
 
Nahoru Odpovědět 3. října 12:23
Avatar
jan.pichl
Člen
Avatar
Odpovídá na Marian Benčat
jan.pichl:

Tak při použití :

var x = (sender as Button).DataContext as Presentation_Container;

            if (x != null)
            {
                SetPropertyValue(x, u => u.MinTimeToChange, 5);

Pokud posílám propertu string to je ok když int tak to nedopadne, protože metoda očekává MemberExpression a v tomto případě je UnaryExpression. Ale se strejdou Googlem jsem za předcházející rady spáchal následující.

  public static void ChangeElement<T>(T target, Expression<Func<T, object>> member, object value)
        {
        var prop = GetMemberInfo(member);

            typeof(T).GetProperty(prop.Name).SetValue(target, value);
}

  public static MemberInfo GetMemberInfo(System.Linq.Expressions.Expression expression)
            {
                var lambda = (LambdaExpression)expression;

                MemberExpression memberExpression;
                if (lambda.Body is UnaryExpression)
                {
                    var unaryExpression = (UnaryExpression)lambda.Body;
                    memberExpression = (MemberExpression)unaryExpression.Operand;
                }
                else
                    memberExpression = (MemberExpression)lambda.Body;

                return memberExpression.Member;
            }

A to se tváří že funguje.

 
Nahoru Odpovědět 3. října 17:48
Děláme co je v našich silách, aby byly zdejší diskuze co nejkvalitnější. Proto do nich také mohou přispívat pouze registrovaní členové. Pro zapojení do diskuze se přihlas. Pokud ještě nemáš účet, zaregistruj se, je to zdarma.

Zobrazeno 5 zpráv z 5.