AGB  ·  Datenschutz  ·  Impressum  







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

RTTI-Fehler bei DSharp

Ein Thema von Codewalker · begonnen am 10. Feb 2013 · letzter Beitrag vom 11. Feb 2013
Antwort Antwort
Benutzerbild von Codewalker
Codewalker

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

RTTI-Fehler bei DSharp

  Alt 10. Feb 2013, 14:15
Hallo zusammen.

Ich nutze schon seit längerem DSharps TEvent<T> bzw. IEvent<T> um mir eine Art Observer zu basteln. Das hat bisher gut geklappt. Bei einer Logging-Funktion bekomme ich aber eine EAssertViolation: "TLogFunc must contain extended RTTI (DSharp.Core.Events.pas:346)". Kann mir jemand weiterhelfen, was das Problem ist. Die Methode sieht eigentlich recht harmlos aus:
Code:
procedure TMGMain.LogFunc(S: string; Args: array of const);
begin
  Logbook.Lines.Add(Format(S, Args));
end;

Geändert von Codewalker (10. Feb 2013 um 14:23 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Codewalker
Codewalker

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

AW: RTTI-Fehler bei DSharp

  Alt 10. Feb 2013, 14:23
Arrrgh, manchmal .. wirklich manchmal ...

Für alle die über sowas stolpern: Für
Code:
array of const
- Parameter (wie man sie an die Format-Funktion füttert) werden keine RTTI-Daten erzeugt. Ergo kann ich diese nicht mit DSharp nutzen. Es gibt natürlich einen QC-Eintrag dafür, was aber auch nicht weiterhilft (ich nutze XE2 - da wird sicher nix gepatcht. Wäre ja zu schön). QualityCentral-Link

Wenn jemand eine andere Art kennt, die man ähnlich leicht benutzen kann für den Aufruf, bin ich ganz Ohr.

Edit:
Code:
array of TVarRec
geht auch nicht (entspricht ja mehr oder weniger dem internen
Code:
array of const

Geändert von Codewalker (10. Feb 2013 um 14:27 Uhr)
  Mit Zitat antworten Zitat
BlackSeven

Registriert seit: 25. Sep 2004
79 Beiträge
 
Delphi XE7 Professional
 
#3

AW: RTTI-Fehler bei DSharp

  Alt 10. Feb 2013, 18:02
Ich würde das wie folgt lösen:

Delphi-Quellcode:
unit Unit1;

interface

uses
  Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
  Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.StdCtrls, RTTI, DSharp.Core.Events,
  DSharp.Core.Reflection;

type

{$M+}
  TTestEvent = reference to procedure(S: string; const Args: TArray<TValue>);
{$M-}

  TForm1 = class(TForm)
    Button1: TButton;
    Memo1: TMemo;
    procedure Button1Click(Sender: TObject);
  private
    FTestEvent: Event<TTestEvent>;
  public
    { Public declarations }
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

function Format(const Format: string; const Args: array of TValue): string;
begin

  Result := System.SysUtils.Format(Format, TValue.ToVarRecs(Args));

end;

procedure LogMessage(const AText: string; const AValues: array of TValue);
var
  Values: TArray<TValue>;

begin

  Form1.FTestEvent.Add(
    procedure(S: string; const Args: TArray<TValue>)
    var
      LMessage: string;

    begin

      LMessage := Format(S, Args);

      Form1.Memo1.Lines.Add(LMessage);

    end);

  Values := TArrayHelper.Copy<TValue>(AValues);

  Form1.FTestEvent.Invoke(AText, Values);

end;

procedure TForm1.Button1Click(Sender: TObject);
begin

  LogMessage('message: %s %d', ['test', 1234]);

end;

end.
  Mit Zitat antworten Zitat
Benutzerbild von Codewalker
Codewalker

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

AW: RTTI-Fehler bei DSharp

  Alt 10. Feb 2013, 20:44
Sehr schöne Lösung - genau so werde ich es wohl lösen.
Danke (schade, dass man Beiträge nicht upvoten kann )
  Mit Zitat antworten Zitat
Benutzerbild von Stevie
Stevie

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

AW: RTTI-Fehler bei DSharp

  Alt 11. Feb 2013, 16:11
Seit r595 kannst du auch folgendes machen:

Delphi-Quellcode:
type
  TLogEvent = class(TEventBase<TLogFunc>)
  protected
    function GetInvoke: TLogFunc; override; // optional
    procedure InitInvoke; override;
    procedure InternalInvoke(s: string; Args: array of const);
  end;

function TLogEvent.GetInvoke: TLogFunc;
begin
  // durch das Überschreiben dieser Methode gewinnst du etwas Performance,
  // da nicht mehr die generische Implementierung benutzt werden muss
  Result := InternalInvoke;
end;

procedure TLogEvent.InitInvoke;
begin
  SetInvoke(@TLogEvent.InternalInvoke);
end;

procedure TLogEvent.InternalInvoke(s: string; Args: array of const);
var
  i: Integer;
begin
  if FEnabled then
    for i := 0 to Count - 1 do
      Handler[i](s, Args);
end;
Du musst dann allerdings die Instanz selber erstellen und kannst nicht das automatische Initialisieren über den Recordtyp Event<T> benutzen.

Für die Zukunft ist geplant, dass man eigene Klassen registrieren kann, welche dann bei Benutzung des Recordtyps erstellt werden (im Event<T>.Initialize ).
Somit kann man für bestimmte Signaturen, selbst eine Event Klasse erstellen, die dann auch über weniger Overhead durch die generische Implementierung verfügt und somit performanter ist.
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight

Geändert von Stevie (11. Feb 2013 um 16:16 Uhr)
  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 12:26 Uhr.
Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz