AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

Observer-Pattern

Ein Thema von Codewalker · begonnen am 15. Dez 2011 · letzter Beitrag vom 16. Nov 2012
Antwort Antwort
Benutzerbild von Codewalker
Codewalker

Registriert seit: 18. Nov 2005
Ort: Ratingen
945 Beiträge
 
Delphi XE2 Professional
 
#1

AW: Observer-Pattern

  Alt 16. Nov 2012, 13:40
*Thread ausgrab*

Ich bin mit dem Pattern immer noch nicht wirklich zufrieden. Im Prinzip hätte ich gerne einen Observer, der für alle Events herhalten kann. Ich habe jetzt dazu mit der RTTI ein wenig gebastelt.
Der Ansatz sieht so aus:
Delphi-Quellcode:
unit thObserver;

interface

uses System.Generics.Collections, Rtti;

type
  TObseverItem = class
  public
    Methods: TList<TRttiMethod>;
    constructor Create;
    destructor Destroy; override;
  end;

  TObserver = class(TDictionary<string, TObseverItem>)
  public
    procedure RegisterObserver(EventName: string; Method: TRttiMethod);
    procedure UnregisterObserver(EventName: string; Method: TRttiMethod);
    procedure Call(EventName: string; Args: array of TValue);
  end;

implementation

procedure TObserver.Call(EventName: string; Args: array of TValue);
var
  Item: TRttiMethod;
begin
  for Item in Items[EventName].Methods do
  begin
    Item.Invoke(Item, Args)
  end;
end;

procedure TObserver.RegisterObserver(EventName: string;
  Method: TRttiMethod);
begin
  if not ContainsKey(EventName) then
  begin
    Add(EventName, TObseverItem.Create);
  end;
  Items[EventName].Methods.Add(Method);
end;

procedure TObserver.UnregisterObserver(EventName: string;
  Method: TRttiMethod);
begin
  Items[EventName].Methods.Remove(Method);

  if Items[EventName].Methods.Count = 0 then
  begin
    Items[EventName].Free;
    Remove(EventName);
  end;
end;

{ TObseverItem }

constructor TObseverItem.Create;
begin
  Methods := TList<TRttiMethod>.Create;
end;

destructor TObseverItem.Destroy;
begin
  Methods.Free;
  inherited;
end;

end.
Es läuft aber noch nicht rund. Im Moment sieht das ganze im Aufruf so aus:

Delphi-Quellcode:
type
  TMyClass = class
  private
    FObserver: TObserver;
  public
    constructor Create;
    destructor Destroy; override;

    procedure RegisterOnChange(Method: TRttiMethod);
    procedure UnregisterOnChange(Method: TRttiMethod);
  end;

{...}

// zum auslösen, um z.B. Button1 als Parameter zu übergeben (wir gehen davon aus, dass die registrierte Funktion auch darauf passt
   FObserver.Call('OnChange', [TValue.From<TButton>(Button1)]);
Um jetzt von außerhalb mich an das Event anzuhängen, mache ich folgendes:

Delphi-Quellcode:

procedure TForm1.TestOnChange(Button: TButton);
begin
// ...
end;

{...}

var
  Method: TRttiMethod;
  context: TRttiContext;
begin
  Method := context.GetType(Self.ClassType).GetMethod('TestOnChange');

  Myclass.RegisterOnAquireAchievement(Method);
Das ganze geht so lange gut, bis der Aufruf über das Invoke erfolgt, dann hagelt es eine AccessViolation. Ist das so, wie ich das vorhabe überhaupt möglich? (Noch schöner wäre, wenn man sich mit Include und Exclude an das Event anhängen könnte, ähnlich wie in .NET. Also dann sowas wie
Myclass.OnChange.Include(MeinEventHandler) , aber da weiß ich auch nicht, ob und wie das geht).
  Mit Zitat antworten Zitat
Benutzerbild von Sir Rufo
Sir Rufo

Registriert seit: 5. Jan 2005
Ort: Stadthagen
9.454 Beiträge
 
Delphi 10 Seattle Enterprise
 
#2

AW: Observer-Pattern

  Alt 16. Nov 2012, 13:46
Ja geht, das hat Stevie vor kurzem hier gezeigt auch zum Thema Observer

Da is http://www.delphipraxis.net/1190831-post5.html
Kaum macht man's richtig - schon funktioniert's
Zertifikat: Sir Rufo (Fingerprint: ‎ea 0a 4c 14 0d b6 3a a4 c1 c5 b9 dc 90 9d f0 e9 de 13 da 60)
  Mit Zitat antworten Zitat
Benutzerbild von Codewalker
Codewalker

Registriert seit: 18. Nov 2005
Ort: Ratingen
945 Beiträge
 
Delphi XE2 Professional
 
#3

AW: Observer-Pattern

  Alt 16. Nov 2012, 13:56
OKay, das mit dem .Add etc. sieht da gut aus. Nur ist es in seinem Fall ein konkreter Observer. Ich möchte das als wiederverwendbare Klasse machen. Wenn ich es als generische Klasse mache TObserver<T> = class kann ich T ja nicht aufrufen, weil nicht klar ist, dass es eine Methode ist. Und ein Constraint, dass da nur Methodentypen reindürfen kenne ich leider nicht.
Ein Idee wie ich das umschiffe? Dann könnte ich das mit Stevies Lösung verbinden und wäre glücklich
  Mit Zitat antworten Zitat
Benutzerbild von Stevie
Stevie

Registriert seit: 12. Aug 2003
Ort: Soest
4.045 Beiträge
 
Delphi 10.1 Berlin Enterprise
 
#4

AW: Observer-Pattern

  Alt 16. Nov 2012, 13:59
OKay, das mit dem .Add etc. sieht da gut aus. Nur ist es in seinem Fall ein konkreter Observer. Ich möchte das als wiederverwendbare Klasse machen. Wenn ich es als generische Klasse mache TObserver<T> = class kann ich T ja nicht aufrufen, weil nicht klar ist, dass es eine Methode ist. Und ein Constraint, dass da nur Methodentypen reindürfen kenne ich leider nicht.
Ein Idee wie ich das umschiffe? Dann könnte ich das mit Stevies Lösung verbinden und wäre glücklich
TEvent<T> wirft zur Laufzeit eine Exception, wenn T kein Event oder Delegate Typ mit Rtti ist.
Dadurch, dass die Eigenschaft Invoke von T ist, kann man die auch aufrufen (z.B. e.Invoke(Button1); ).
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight

Geändert von Stevie (16. Nov 2012 um 14:02 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Codewalker
Codewalker

Registriert seit: 18. Nov 2005
Ort: Ratingen
945 Beiträge
 
Delphi XE2 Professional
 
#5

AW: Observer-Pattern

  Alt 16. Nov 2012, 14:04
Aber ich scheitere doch schon zur DesignTime, weil ich die Notify-Methode nicht programmieren kann.

Delphi-Quellcode:
unit thObserver;

interface

uses
   System.Generics.Collections, Rtti;

type
  TObserver<T> = class
  private
    FEvents: TList<T>;
  public
    constructor Create;
    destructor Destroy; override;

    procedure Notify(Args: array of TValue);
    property OnEvent: TList<T> read FEvents;
  end;

implementation


{ TObserver<T> }

constructor TObserver<T>.Create;
begin
  FEvents := TList<T>.Create;
end;

destructor TObserver<T>.Destroy;
begin
  FEvents.Free;
  inherited;
end;

procedure TObserver<T>.Notify(Args: array of TValue);
var
  Event: T;
begin
  for Event in FEvents do
  begin
    Event(Args); /// Bis hierhin ist noch alles gut, aber das hier geht ja nicht
  end;
end;

end.
  Mit Zitat antworten Zitat
Benutzerbild von Stevie
Stevie

Registriert seit: 12. Aug 2003
Ort: Soest
4.045 Beiträge
 
Delphi 10.1 Berlin Enterprise
 
#6

AW: Observer-Pattern

  Alt 16. Nov 2012, 14:07
Aber ich scheitere doch schon zur DesignTime, weil ich die Notify-Methode nicht programmieren kann.
Schau dir einfach DSharp.Core.Events.pas an.
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight
  Mit Zitat antworten Zitat
Antwort Antwort


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 18:16 Uhr.
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz