Skip to content

Latest commit

 

History

History
731 lines (572 loc) · 24.9 KB

22_Events.md

File metadata and controls

731 lines (572 loc) · 24.9 KB

LiaScript

Events

Parameter Kursinformationen
Veranstaltung: Vorlesung Softwareentwicklung
Teil: 22/27
Semester @config.semester
Hochschule: @config.university
Inhalte: @comment
Link auf den GitHub: https://github.com/TUBAF-IfI-LiaScript/VL_Softwareentwicklung/blob/master/22_Events.md
Autoren @author


Allgemeine Hinweise

0. Kennen Sie Ihren Editor und dessen Shortcuts!

https://code.visualstudio.com/shortcuts/keyboard-shortcuts-windows.pdf

1. Verwenden Sie einen Code Formater, der Ihnen bei der Restrukturierung Ihres Codes hilft!

2. Evaluieren Sie die Hinweise der Code Analyse sorgfältig, entwerfen Sie ggf. eigene Regeln.

https://learn.microsoft.com/de-de/dotnet/fundamentals/code-analysis/overview?tabs=net-8

Nachgefragt

In der letzten Veranstaltung fragte einer von Ihnen wie der selektive Zugriff auf die MultiCastDelegaten realisieren kann. Zur Erinnerung MulticastDelegate verfügt über eine verknüpfte Liste von Delegaten, die als Aufruf Liste bezeichnet wird und aus einem oder mehreren-Elementen besteht. Wenn ein Multicast Delegat aufgerufen wird, werden die Delegaten in der Aufruf Liste synchron in der Reihenfolge aufgerufen, in der Sie angezeigt werden.


Mulitcast Invocation List

+----------+----------+----------+----------+----------+----------+
| Add      | Multiply | Multiply | Divide   | Add      | Divide   |
+----------+----------+----------+----------+----------+----------+                               .

Frage: Wie können wir unsere MultiCastDelegaten verwalten und selektiv auf einzelne Elemente zugreifen?

vgl. https://learn.microsoft.com/en-us/dotnet/api/system.delegate.getinvocationlist?view=net-8.0

using System;
using System.Reflection;
using System.Collections.Generic;

public class Program
{
  public delegate int Calc(int x, int y);

  static int Add(int x, int y){
      Console.WriteLine("x + y");
      return x + y;
  }

  static int Multiply(int x, int y){
      Console.WriteLine("x * y");
      return x * y;
  }

  static int Divide(int x, int y){
      Console.WriteLine("x / y");
      return x / y;
  }

  public static void Main(string[] args){
    Calc computer = Add;
    computer += Multiply;
    computer += Multiply;
    computer += Divide;
    computer += Add;
    computer += Divide;
    computer -= Add;
    Console.WriteLine("Zahl von eingebundenen Delegates {0}",
                      computer.GetInvocationList().GetLength(0));

    // Individueller Aufruf der einzelnen Einträge
    var x = computer.GetInvocationList();
    Console.WriteLine("Typ der Invocation List {0}", x.GetType());
    Console.WriteLine(x[0].DynamicInvoke(1,2));
    Console.WriteLine(x[1].DynamicInvoke(3,5));   

    // Übergreifender Aufruf aller Einträge                     
    Console.WriteLine(computer(40,8));
  }
}

@LIA.evalWithDebug(["main.cs"], mcs main.cs, mono main.exe)

Frage: Wie ließe sich das Codebeispiel verbessern? Wie war das gleich noch Func oder Action?

Wiederholung

Wie war das noch mal, welche Elemente (Member) zeichnen einen Klasse unter C# aus?

                                  {{0-1}}

| Bezeichnung | | Konstanten | | Felder | | Methoden | | Eigenschaften | | Indexer | | Ereignisse | | Operatoren | | Konstruktoren | | Finalizer | | Typen |


{{1}}


| Bezeichnung | Bedeutung | | Konstanten | Konstante Werte innerhalb einer Klasse | | Felder | Variablen der Klasse | | Methoden | Funktionen, die der Klasse zugeordnet sind | | Eigenschaften | Mechanismen zum Lesen und Schreiben auf geschützter Variablen | | Indexer | Spezifikation eines Indexoperators für die Klasse | | Ereignisse | ? | | Operatoren | Definition von eigenen Symbolen für die Arbeit mit Instanzen der Klasse | | Konstruktoren | Bündelung von Aktionen zur Initialisierung der Klasseninstanzen | | Finalizer | Aktionen, die vor dem "Löschen" der Instanz ausgeführt werden | | Typen | Geschachtelte Typen, die innerhalb der Klasse definiert werden |


Motivation und Idee der Events

Was haben wir mit den Delegaten erreicht? Wir sind in der Lage aus einer Klasse, auf Methoden (einer anderer Klassen) zurückzugreifen, über die wir per Referenz informiert wurden. Die aufgerufene Methode wird der aufrufenden Klasse über einen Delegaten bekannt gegeben. Es erfolgt eine Typprüfung der Parameter.

Damit sind die beiden Klassen nur über eine Funktionssignatur miteinander "lose" gekoppelt. Welche Konzepte lassen sich damit umsetzen?

Publish-Subscribe Prinzip

OOPGeschichte

Publish-Subscribe (Pub / Sub) ist ein Nachrichtenmuster, bei dem Publisher Nachrichten an Abonnenten senden. In der Softwarearchitektur bietet Pub / Sub-Messaging Ereignisbenachrichtigungen für verteilte Anwendungen, insbesondere für Anwendungen, die in kleinere, unabhängige Bausteine ​​entkoppelt sind.

Das Publish/Subscribe-Nachrichtenmodell hat folgende Vorteile:

  • Der Publisher braucht gar nicht zu wissen, wer Subscriber ist. Es erfolgt keine explizite Adressierung
  • Der Subscriber ist vom Publisher entkoppelt, er empfängt nur die Nachrichten, die für ihn auch relevant sind.
  • Der Subscriber kann sich jederzeit aus der Kommunikation zurückziehen oder ein anderes Topic subskribieren. Auf den Publisher hat das keine Auswirkung.
  • Damit ist die Messaging-Topologie dynamisch und flexibel. Zur Laufzeit können neue Subscriber hinzukommen.

C# etabliert für die Nutzung der Pub-Sub Kommunikation Events. Dies sind spezielle Delegate-Variablen, die mit den Schlüsselwort event als Felder von Klassen deklariert werden.

Events in C#

{{0-1}}


Der Publisher ist eine Klasse, die ein Delegaten enthält. Der Publisher entscheidet damit darüber, wann Nachrichten versandt werden. Auf der anderen Seite finden sich die Subscriber-Methoden, die ausgehend vom aktivierten Delegaten im Publisher zur Ausführung kommen. Ein Subscriber hat keine Kenntnis von anderen Subscribern. Events sind ein Feature aus C# dass dieses Pattern formalisiert.

Merke: Ein Event ist ein Klassenmember, dass die Features des Delegatenkonzepts nutzt, um eine Publisher-Subscribe Interaktion zu realisieren.

Im einfachsten Fall lässt sich das Event-Konzept folgendermaßen anwenden:

// Schritt 1
// Wir definieren einen Delegat, der das Format der Subscriber Methoden
// (callbacks) spezifiziert - in diesem Beispiel ohne Parameter
// Hier wird ein nicht-generischer Handler genutzt.
public delegate void varAChangedHandler();

// Schritt 2
// Wir integrieren ein event in unser Publisher Klasse, dass den Delegaten
// abbildet
public class Publisher{
  public event varAChangedHandler OnAChangedHandler;

  // Schritt 3
  // Wir implementieren das "Feuern" des Events
  public magicMethod(){
     if (oldA != newA) OnAChangedHandler();
  }
}

// Schritt 4
// Implementieren des Subscribers - in diesem Fall wurde eine separate Klasse
// gewählt. 
public class Subscriber{
  public static void m_OnPropertyChanged(){
      Console.WriteLine("A was changed!");
  }
}

// Schritt 5
// "Einhängen" der Subscriber Methode in den Publisher-Delegate
public static void Main(string[] args){
  Publisher myPub = new Publisher();
  Subscriber mySub = new Subscriber();
  myPub.OnAChangedHandler += new varAChangedHandler(mySub.m_OnPropertyChanged);
}

Welche Konsequenzen ergeben sich daraus?

  • Der Publisher bestimmt, wann ein Ereignis ausgelöst wird. Die Subscriber bestimmen, welche Aktion als Reaktion auf das Ereignis ausgeführt wird.

  • Es ist eine 1:n Relation. Ein Ereignis entstammt einem Publisher, kann aber mehrere Subscriber adressieren.

  • Ereignisse, die keine Subscriber haben, werden nie ausgelöst. Das bedeutet, dass ein Publisher die Subscriber kennen muss.

  • Ereignisse werden in der Regel verwendet, um Benutzeraktionen wie Mausklicks oder Menüauswahlen in GUI-Schnittstellen zu signalisieren.

  • In der .NET Framework-Klassenbibliothek basieren Ereignisse auf dem EventHandler-Delegaten und der EventArgs-Basisklasse.


                               {{1-2}}

Was passiert hinter den Kulissen?

Wenn wir ein event deklarieren

public class Publisher{
  public event VarAChangedHandler AChanged;
}

passieren folgende 3 Dinge:

  1. Der Complier erzeugt einen privaten Delegaten mit sogenannten Event Accessoren (add und remove).
VarAChangedHandler aChanged; // private Delegate
public event VarAChangedHandler AChanged
{
  add {aChanged += value;}
  remove {aChanged -= value;}
}
  1. Der Compiler sucht innerhalb der Publisher Klasse nach Referenzen, die auf AChanged und lenkt diese auf das private Delegate um.
  2. Der Compiler mappt alle += Operationen außerhalb auf die Zugriffsmethoden add and remove.

                              {{2-3}}

Ok, nett, aber das würde ich gern an einem Beispiel sehen!

Beispiel 1

Das Beispiel vereinfacht das Vorgehen, in dem es Publisher und Subscriber in einer Funktion zusammenfasst. Damit kann auf das Event uneingeschränkt zurückgegriffen werden. Dazu gehört auch, dass das Event mit Invoke ausgelöst wird.

using System;
using System.Reflection;
using System.Collections.Generic;

public delegate void DelEventHandler();

class Program
{
    public static event DelEventHandler myEvent;
    public static void Main(string[] args){
        myEvent += new DelEventHandler(Fak1);
        myEvent += Fak2;
        myEvent += () => {Console.WriteLine("Fakultät 3");};
        myEvent.Invoke();
    }

    static void Fak1()
    {
        Console.WriteLine("Fakultät 1");
    }

    static void Fak2()
    {
        Console.WriteLine("Fakultät 2");
    }
}

@LIA.evalWithDebug(["main.cs"], mcs main.cs, mono main.exe)


                              {{3-4}}

Beispiel 2

using System;
using System.Reflection;
using System.Collections.Generic;

public delegate void DelPriceChangedHandler();

public class Stock{
    decimal price = 5;
    public event DelPriceChangedHandler OnPropertyPriceChanged;
    public decimal Price{
      get { return price; }
      set { if (price != value){
              if (OnPropertyPriceChanged != null){
                 OnPropertyPriceChanged();
                 price = value;
              }
            }
          }
    }
}

public class MailService{
  public static void stock_OnPropertyChanged(){
      Console.WriteLine("MAIL - Price of stock was changed!");
  }
}

public class Logging{
  public static void stock_OnPropertyChanged(){
      Console.WriteLine("LOG  -  Price of stock was changed!");
  }
}

class Program {
    public static void Main(string[] args){
        Stock GoogleStock = new Stock();
        GoogleStock.OnPropertyPriceChanged += MailService.stock_OnPropertyChanged;
        GoogleStock.OnPropertyPriceChanged += Logging.stock_OnPropertyChanged;
        Console.WriteLine("We change the stock price now!");
        GoogleStock.Price = 10;
    }
}

@LIA.eval(["main.cs"], mcs main.cs, mono main.exe)

**Ja, aber ... ** Was unterscheidet eine Delegate von einem Event? Was würde passieren, wenn wir das Key-Wort aus dem Code entfernen?

Die Implementierung wäre nicht so robust, da folgende Möglichkeiten offen ständen:

Eingriff Bedeutung Verhindert
GoogleStock.OnPropertyPriceChanged = null; Löscht alle Callback-Handler ja
GoogleStock.OnPropertyPriceChanged = DelPriceChangedHandler(MailService.stock_OnPropertyChanged); Setzt einen einzigen Handler (und löscht alle anderen) ja
GoogleStock.OnPropertyPriceChanged.Invoke(); Auslösen des Events innerhalb eines Subscribers ja

Was fehlt Ihnen an der Implementierung?

Richtig, die Möglichkeit auf die Daten zurückzugreifen.


Events - Praktische Implementierung

Die Möglichkeit Parameter strukturiert und wiederverwendbar zu übergeben und entsprechend generische EventHandler zu integrieren.

Parameter

Welche Informationen sollten an die Subscriber weitergereicht werden? Zum einen der auslösende Publisher (Wer ist verantwortlich?) und ggf. weitere Daten zum Event (Warum ist die Information eingetroffen?).

Im Beispiel konzentrieren wir uns auf die Default-Delegates, die Bestandteil der .NET Umgebung ist

Delegate Aufruf Link
EventHandler Delegate void EventHandler(object sender, EventArgs e) Link
EventHandler<TEventArgs> Delegat EventHandler<TEventArgs>(object? sender, TEventArgs e); Link
using System;
using System.Reflection;
using System.Collections.Generic;

//Brauchen wir nicht mehr, vielmehr wird auf einen vordefinierten EventHandler
//zurückgegriffen.
//public delegate void DelPriceChangedHandler();
public class Stock{
    decimal price = 5;
    // Hier ersetzen wir unser Delegate "DelPriceChangedHandler" durch den
    // Standard-Typ EventHandler
    public event EventHandler OnPropertyPriceChanged;
    public decimal Price{
      get { return price; }
      set { if (price != value){
              if (OnPropertyPriceChanged != null){
                 OnPropertyPriceChanged(this, EventArgs.Empty );
                 price = value;
              }
            }
          }
    }
}

public class MailService{
  public static void stock_OnPropertyChanged(object sender, EventArgs e){
      Console.WriteLine("MAIL - Price of stock was changed!");
      Console.WriteLine("Wer ruft? - {0}", sender);
      Console.WriteLine("Werte? - {0}", e);
  }
}

class Program {
    public static void Main(){
        Stock GoogleStock = new Stock();
        GoogleStock.OnPropertyPriceChanged += new
                EventHandler(MailService.stock_OnPropertyChanged);
        Console.WriteLine("We changed the stock price now!");
        GoogleStock.Price = 10;
    }
}

@LIA.eval(["main.cs"], mcs main.cs, mono main.exe)

Generic Events

Nun wollen wir einen Schritt weitergehen und spezifischere Informationen mit dem Event an die Subscriber weiterreichen. Dafür implementieren wir eine von EventArgs abgeleitete Klasse.

public class PriceChangedEventArgs : EventArgs
{
    public decimal Difference { get; set; }
}

Diese soll den Unterschied zwischen altem und neuen Preis umfassen, damit der Subscriber die Relevanz der Information beurteilen kann.

Damit brauchen wir aber auch ein neues Delegate für unser nun nicht mehr Standard Event

void EventHandler(object sender, PriceChangedEventArgs e)

Um diese Anpassungen beim Datentyp zu realisieren existiert bereits eine generischen Form von EventHandler mit der Signatur

public delegate void EventHandler<TEventArgs>(object source,
                                              TEventArgs e)
                                              where TEventArgs: EventArgs;
using System;
using System.Reflection;
using System.Collections.Generic;

public class Stock{
    decimal price = 5;
    public event EventHandler<PriceChangedEventArgs> OnPropertyPriceChanged;
    public decimal Price{
      get { return price; }
      set { if (price != value){
              if (OnPropertyPriceChanged != null){
                 PriceChangedEventArgs myEventArgs = new PriceChangedEventArgs();
                 myEventArgs.Difference = price - value;
                 OnPropertyPriceChanged(this, myEventArgs );
                 price = value;
              }
            }
          }
    }
}

public class PriceChangedEventArgs : EventArgs
{
    public decimal Difference { get; set; }
}

public class MailService{
  public static void stock_OnPropertyChanged(object sender, PriceChangedEventArgs e){
      Console.WriteLine("MAIL - Price of stock was changed!");
      Console.WriteLine("Wer ruft? - {0}", sender);
      Console.WriteLine("Preisänderung? - {0}", e.Difference);
  }
}

class Program {
    public static void Main(string[] args){
        Stock GoogleStock = new Stock();
        GoogleStock.OnPropertyPriceChanged += new
                EventHandler<PriceChangedEventArgs>(MailService.stock_OnPropertyChanged);
        Console.WriteLine("We manipulate the stock price now!");
        GoogleStock.Price = 10;
    }
}

@LIA.eval(["main.cs"], mcs main.cs, mono main.exe)

Events und Ausnahmebehandlung

                      {{0-1}}

Welche Rückmeldung hätten Sie mit Blick auf die Flexibilität des Codes an einen Mitstreiter?

using System;
using System.Reflection;
using System.Collections.Generic;

public delegate void MyEventHandler(string message);

public class Publisher
{
    public event MyEventHandler MyEvent;
    public void RaiseEvent(string message)
    {
        MyEvent?.Invoke(message);
    }
}

public class Subscriber1
{
    //int x = 100;
    //int divider = 0;
    public void OnEventRaised(string message)
    {
        Console.WriteLine($"Sub 1 - Event received: {message}");
        //Console.WriteLine($"Sub 1 - Event received: {x / divider}");
    }
}

public class Subscriber2
{
    public void OnEventRaised(string message)
    {
        Console.WriteLine($"Sub 2 - Event received: {message}");
    }
}

class Program {
    public static void Main(string[] args){
        var publisher = new Publisher();
        var subscriber1 = new Subscriber1();
        publisher.MyEvent += subscriber1.OnEventRaised;
        var subscriber2 = new Subscriber2();
        publisher.MyEvent += subscriber2.OnEventRaised;
        publisher.RaiseEvent("Hallo Welt");
    }
}

@LIA.evalWithDebug(["main.cs"], mcs main.cs, mono main.exe)


{{1-2}}

using System;
using System.Reflection;
using System.Collections.Generic;

// nicht mehr benötigt !
// public delegate void MyEventHandler(string message);

public class CustomEventArgs : EventArgs
{
    public string Message { get; }
    public CustomEventArgs(string msg)
    {
        Message = msg;
    }
}

public class Publisher
{
    public event EventHandler<CustomEventArgs> MyEvent;
    public void RaiseEvent(string message)
    {
	    foreach (var handler in MyEvent.GetInvocationList())
	    {
	        try
	        {
	            handler.DynamicInvoke(this, new CustomEventArgs(message));
	        }
	        catch (Exception ex)
	        {
	            Console.WriteLine($"Exception caught: {ex.Message}");
	        }
	    }
    }
}

public class Subscriber1
{
    int x = 100;
    int divider = 0;
    public void OnEventRaised(object sender, CustomEventArgs e)
    {
        Console.WriteLine($"Sub 1 - Event received: {e.Message}");
        Console.WriteLine($"Sub 1 - Event received: {x/divider}");
    }
}

public class Subscriber2
{
    public void OnEventRaised(object sender, CustomEventArgs e)
    {
        Console.WriteLine($"Sub 1 - Event received: {e.Message}");
    }
}

class Program {
    public static void Main(string[] args){
        var publisher = new Publisher();
        var subscriber1 = new Subscriber1();
        publisher.MyEvent += subscriber1.OnEventRaised;
        var subscriber2 = new Subscriber2();
        publisher.MyEvent += subscriber2.OnEventRaised;
        publisher.RaiseEvent("Hallo Welt");
    }
}

@LIA.evalWithDebug(["main.cs"], mcs main.cs, mono main.exe)

Anpassung der Subscribe/Unsubscribe Methoden

In spezifischen Fällen kann es notwendig sein, die Subscriber-Methoden zu adaptieren.

private MyEventHandler _myEvent;

public event MyEventHandler MyEvent
{
    add
    {
        Console.WriteLine("Subscriber added");
        _myEvent += value;
    }
    remove
    {
        Console.WriteLine("Subscriber removed");
        _myEvent -= value;
    }
}

Anonyme / Lambda Funktionen als Subscriber

using System;
using System.Reflection;
using System.Collections.Generic;

public class CustomEventArgs : EventArgs
{
    public string Message { get; }
    public CustomEventArgs(string msg)
    {
        Message = msg;
    }
}

public class Publisher
{
    public event EventHandler<CustomEventArgs> MyEvent;
    public void RaiseEvent(string message)
    {
        MyEvent?.Invoke(this, new CustomEventArgs(message));
    }
}

class Program {
    public static void Main(){
        var publisher = new Publisher();

        // Using an anonymous method
        publisher.MyEvent += delegate(object sender, CustomEventArgs args) 
        {
            Console.WriteLine($"Anonyme Method received: {args.Message}");
        };

        // Or using a lambda expression
        publisher.MyEvent += (sender, args) => Console.WriteLine($"Lambda expression received: {args.Message}");
        publisher.RaiseEvent("Hallo Welt");
    }
}

@LIA.evalWithDebug(["main.cs"], mcs main.cs, mono main.exe)

Eine sehr anschauliche Darstellung dazu findet sich unter https://gehirnwindung.de/categories/csharp/tanz-den-lambda-mit-mir

Grafische Nutzer Interfaces

siehe Codebeispiel wpf_sharp im Projektordner

BlockedGUI