V 1.0 Szabó Zsolt, Óbudai Egyetem, Haladó Programozás Eseménykezelés ismétlés Névtelen metódusok (anonymous methods)
V 1.0 Szabó Zsolt, Óbudai Egyetem, Haladó Programozás Eseménykezelés ismétlés Névtelen metódusok (anonymous methods)
V 1.0 Szabó Zsolt, Óbudai Egyetem, PPT és VEP tananyag A prezentációban eseményt kiváltó és eseményt feldolgozó osztályról beszélünk, de eseménykezelés lehetséges akár egyetlen osztályon belül is Cél: esemény bekövetkezésének jelzése más modul/osztály részére Módszerek: Függvénymutató átadása (DOS), Delegate-ek használata (C#), Interfészek használata (JAVA) Ezen az órán csak a C# megvalósítással foglalkozunk Eseménykezelés
V 1.0 Szabó Zsolt, Óbudai Egyetem, Egy típus, amelybe metódusokat tudunk „belerakni” A deklarációnál meg kell kötni, hogy milyen szignatúrájú metódust tud tárolni az adott delegate: delegate double MyDelegate(int param1, string param2); –Név: MyDelegate –Visszatérési típus: double –Paraméterek: 2 db, int+string Fontos: ez még csak egy típusdeklaráció –Változó nélkül nem használható –Referenciatípus –Metóduson belül nem deklarálható –Osztályon belül és kívül is deklarálható Delegate
V 1.0 Szabó Zsolt, Óbudai Egyetem, delegate double MyDelegate(int param1, string param2); double funct(int elso, string masodik) { return elso + masodik.Length; } Más metóduson belül: MyDelegate del; del = new MyDelegate(funct); Console.WriteLine(del(5, "hello")); Delegate használata
V 1.0 Szabó Zsolt, Óbudai Egyetem, delegate double MyDelegate(int param1, string param2); double funct(int elso, string masodik) { return elso + masodik.Length; } Osztály deklarációs részében (metódusokon kívül): –MyDelegate del; Más metóduson belül: –del += funct; Console.WriteLine(del(5, "hello")); –Amíg nem rakunk a delegate-be metódust, az értéke NULL hívás előtt ellenőrizni kell! –Lehetséges több metódust belerakni egy delegate-be Delegate használata
V 1.0 Szabó Zsolt, Óbudai Egyetem, delegate bool Összehasonlító(object bal, object jobb); class EgyszerűCserésRendező { public static void Rendez(object[] Tömb, Összehasonlító nagyobb) { for (int i = 0; i < Tömb.Length; i++) for (int j = i + 1; j < Tömb.Length; j++) if (nagyobb(Tömb[j], Tömb[i])) { object ideiglenes = Tömb[i]; Tömb[i] = Tömb[j]; Tömb[j] = ideiglenes; } Delegate-paraméter
V 1.0 Szabó Zsolt, Óbudai Egyetem, Delegate-paraméter példaosztály class diak { public string nev; public int kreditek; public diak(string ujnev, int ujkreditek) { nev = ujnev; kreditek = ujkreditek; } public override string ToString() { return nev + " - " + kreditek; } diak[] csoport = new diak[] { new diak("Első Egon", 52), new diak("Második Miksa", 97), new diak("Harmadik Huba", 10), new diak("Negyedik Néró", 89), new diak("Ötödik Ödön", 69) };
V 1.0 Szabó Zsolt, Óbudai Egyetem, Delegate-paraméter kód bool mycomp(object elso, object masodik) { return ((elso as diak).kreditek < (masodik as diak).kreditek); } Összehasonlító del = new Összehasonlító(mycomp); EgyszerűCserésRendező.Rendez(csoport, del); VAGY EgyszerűCserésRendező.Rendez(csoport, mycomp);
V 1.0 Eseménykezelés Szabó Zsolt, Óbudai Egyetem, Eddig: tudunk egy osztályon belül delegate-et deklarálni, és használni Cél: osztályok közötti átjárhatóság funkcionalitás szempontjából a delegate erre jó, a delegate típusnál csak a metódus szignatúrája számít, az azt tartalmazó osztály nem Gyakorlatilag az eseményt kiváltó osztályban egy publikus delegate típusú változó kell Az „event” kulcsszót használjuk annak jelzésére, hogy ez egy esemény lesz –Szerepelhet interface-ben (sima delegate változó nem) –Meghívni csak a tartalmazó osztályból lehet (sima publikus delegate-et bárhonnan) –Kötött szignatúra: void visszatérési típus, Sender + EventArgs paraméter
V 1.0 Szabó Zsolt, Óbudai Egyetem, Az eseményt lekezelő osztályból rakunk bele egy metódust az eseményt kiváltó osztály delegate-jébe (csak += és -= operátorok!) Vagyis, ha meghívjuk a delegate-et a külső metódust hívjuk meg Fontos: –Az eseményt kiváltó osztályban az esemény-delegate meghívásakor nem tudjuk, hogy a delegate tartalmaz-e metódust (vagyis: nem tudjuk, hogy rendeltek-e eseménykezelőt az eseményhez) –Ezért a delegate meghívása előtt mindig figyelni kell a NULL értéket –Lehetőség van egy eseményhez több eseménykezelőt is rendelni, bár ez a Visual Studio tervező nézetében nem megoldott Eseménykezelés
V 1.0 Szabó Zsolt, Óbudai Egyetem, Eseménykezelés – Névkonvenciók FeladatNévElhelyezés EseményparaméterValamiEventArgsA névtérben vagy a kiváltó osztályban DelegateValamiEventHandlerA névtérben vagy a kiváltó osztályban EseményValamiKiváltó osztályban Eseményt közvetlenül meghívó metódus OnValamiKiváltó osztályban Esemény lekezelése---Lekezelő osztályban
V 1.0 Szabó Zsolt, Óbudai Egyetem, Hozzunk létre Windows Forms alkalmazást, benne a korábban említett Diák {Név, Kreditek} osztállyal Legyen lehetőségünk diákok felvitelére, a felvitt diákok jelenjenek meg egy ListBox vezérlőben Az egyes diákoknak gomb segítségével tudjuk növelni a kreditszámát A diák osztályban következzen be a PrímaDiák esemény, ha a kreditszám-változás után a kreditek száma prímszám Az esemény küldője a kiváltó példány legyen, eseményparaméterként küldjük el az aktuális időt A Form osztályban iratkozzunk fel az eseményre, és a prím- diákokról vezessünk egy külön listát Eseménykezelés feladat
V 1.0 Szabó Zsolt, Óbudai Egyetem, Haladó Programozás Eseménykezelés ismétlés Névtelen metódusok (anonymous methods)
V 1.0 Szabó Zsolt, Óbudai Egyetem, Angolul: anonymous methods Célok: –Kód lerövidítése, olvashatóbbá tétele –Apró, „levegőben lógó” metódusok és eseménykezelők számának csökkentése –A metódussal való paraméterezés megkönnyítése, rövidítése –Ezáltal a létrehozott osztályok „általánosságának” növelése (az osztály működése is módosíthatóvá válik azáltal, hogy metódusokat adhatunk meg paraméterül) Névtelen metódusok
V 1.0 Szabó Zsolt, Óbudai Egyetem, Gyakori példa: dinamikus vezérlők létrehozásánál for (int x = 0; x < 5; x++) { for (int y = 0; y < 5; y++) { Button uj = new Button(); uj.Top = y * 50; uj.Left = x * 50; uj.Width = uj.Height = 40; uj.Text = x + ";" + y; Controls.Add(uj); uj.Click += new EventHandler(uj_Click); } } // Cél: Mindegyik gombra fusson le egy eseménykezelő
V 1.0 Szabó Zsolt, Óbudai Egyetem, Eseménykezelő írásának problémái Tipikusan: egy nagyobb programhoz több saját vezérlő, mindegyikben több saját osztály, több egyedi esemény van Probléma: Az események lekezelésekor minden eseménykezelő egy külön metódus, amik sokszor nagyon egyszerűek. Egy nagy programban rengeteg soros metódus kerül, ráadásul az eseménykezelő hozzárendelése (a += operátoros sor) és az eseménykezelő kódja távol kerül egymástól külön deklarált metódusok helyett úgynevezett névtelen metódusokat használunk „anonymous methods” Lényegük: a metódust nem deklaráljuk külön (ezért névtelen: nincs névvel deklarálva, csak kóddal), így egyszerűen megmondjuk, hogy „ha bekövetkezik az esemény, akkor tedd EZT”
V 1.0 Szabó Zsolt, Óbudai Egyetem, Gyakori példa: dinamikus vezérlők létrehozásánál Button uj = new Button(); uj.Top = y * 50; uj.Left = x * 50; uj.Width = uj.Height = 40; uj.Text = x + ";" + y; Controls.Add(uj); uj.Click += delegate { MessageBox.Show("CLICKED"); }; Záró pontosvessző !!!
V 1.0 Szabó Zsolt, Óbudai Egyetem, Gyakori példa: dinamikus vezérlők létrehozásánál Button uj = new Button(); uj.Top = y * 50; uj.Left = x * 50; uj.Width = uj.Height = 40; uj.Text = x + ";" + y; Controls.Add(uj); uj.Click += delegate(object kuldo, EventArgs esemenypar) { MessageBox.Show((kuldo as Button).Text); };
V 1.0 Szabó Zsolt, Óbudai Egyetem, Névtelen metódusok Előny: könnyebben olvasható kód (egyesek szerint ) Csak az egyszer használatos, és lehetőleg rövid eseménykezelőt írjuk így meg: a hosszú kód olvashatatlan, és mivel „beágyazott kód”, ezért nem újrafelhasználható Lehetőleg ne ágyazzunk egymásba névtelen metódusokat Használatukkal lehetőség van leegyszerűsíteni a korábban delegate-paraméterek használatát is
V 1.0 Szabó Zsolt, Óbudai Egyetem, delegate bool Összehasonlító(object bal, object jobb); class EgyszerűCserésRendező { public static void Rendez(object[] Tömb, Összehasonlító nagyobb) { for (int i = 0; i < Tömb.Length; i++) for (int j = i + 1; j < Tömb.Length; j++) if (nagyobb(Tömb[j], Tömb[i])) { object ideiglenes = Tömb[i]; Tömb[i] = Tömb[j]; Tömb[j] = ideiglenes; } Delegate-paraméter
V 1.0 Szabó Zsolt, Óbudai Egyetem, Delegate-paraméter példaosztály class diak { public string nev; public int kreditek; public diak(string ujnev, int ujkreditek) { nev = ujnev; kreditek = ujkreditek; } public override string ToString() { return nev + " - " + kreditek; } diak[] csoport = new diak[] { new diak("Első Egon", 52), new diak("Második Miksa", 95), new diak("Harmadik Huba", 10), new diak("Negyedik Néró", 84), new diak("Ötödik Ödön", 69) };
V 1.0 Szabó Zsolt, Óbudai Egyetem, Delegate-paraméter példa EgyszerűCserésRendező.Rendez(csoport, delegate(object elso, object masodik) { return (elso as diak).nev. CompareTo((masodik as diak).nev) < 0; }); EgyszerűCserésRendező.Rendez(csoport, delegate(object elso, object masodik) { return (elso as diak).kreditek < (masodik as diak).kreditek; });
V 1.0 Szabó Zsolt, Óbudai Egyetem, Hozzunk létre Windows Forms alkalmazást, benne a korábban említett Diák {Név, Kreditek} osztállyal Legyen lehetőségünk diákok felvitelére, a felvitt diákok jelenjenek meg egy ListBox vezérlőben Az egyes diákoknak gomb segítségével tudjuk növelni a kreditszámát A diák osztályban következzen be a Prim esemény, ha a kreditszám-változás után a kreditek száma prímszám Az esemény küldője a kiváltó példány legyen, eseményparaméterként küldjük el az aktuális időt A Form osztályban iratkozzunk fel az eseményre, és a prím- diákokról vezessünk egy külön listát A fenti eseménykezelőt írjuk át névtelen metódussá Anonymous függvények feladat
V 1.0 Szabó Zsolt, Óbudai Egyetem, Haladó Programozás Eseménykezelés ismétlés Névtelen metódusok (anonymous methods)
V 1.0 Szabó Zsolt, Óbudai Egyetem, Források Events tutorial: Event modifier: Anonymous methods: Anonymous methods: Anonymous methods: Anonymous methods: anonymous-methods-vs-formally-defined-methods Delegate, Anonymous, Event: Reiter István: C# jegyzet ( oldal
V 1.0 Szabó Zsolt, Óbudai Egyetem,