Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Algorithmen, Datenstrukturen und Klassendesign (https://www.delphipraxis.net/78-algorithmen-datenstrukturen-und-klassendesign/)
-   -   Funktion über ihren Namen als String aufrufen (https://www.delphipraxis.net/173860-funktion-ueber-ihren-namen-als-string-aufrufen.html)

Jumpy 21. Mär 2013 11:11

Funktion über ihren Namen als String aufrufen
 
Hallo,

in manchem VBA kann man mit einem Funktionsnamen der als String vorliegt, diese Funktion aufrufen:
CALL("MeineFunktion"). Weiß gerade nicht wie/ob man dabei auch Parameter übergeben kann, ist aber auch egal, da ich das nur ohne Parameter brauche.

Die Frage ist natürlich, ob das auch in Delphi irgendwie geht.

P.S.: Ich hab zwar wie ich gerade sehe überall Funktion geschrieben, meinte allerdings Prozedur.:oops:

DeddyH 21. Mär 2013 11:16

AW: Funktion über ihren Namen als String aufrufen
 
Sofern es sich um Methoden handelt, wäre evtl. etwas mit Delphi-Referenz durchsuchenTMethod und Delphi-Referenz durchsuchenMethodAddress zu machen.

Nersgatt 21. Mär 2013 11:17

AW: Funktion über ihren Namen als String aufrufen
 
Mit Deinem Delphi 6 wird das so wohl nicht gehen. Bei neueren Versionen (ich glaub ab 2010) gibt es die RTTI.

mjustin 21. Mär 2013 11:28

AW: Funktion über ihren Namen als String aufrufen
 
Zitat:

Zitat von Nersgatt (Beitrag 1208183)
Mit Deinem Delphi 6 wird das so wohl nicht gehen. Bei neueren Versionen (ich glaub ab 2010) gibt es die RTTI.

Ab 2010 gab es Extended RTTI, davor gab es ('einfache') RTTI, in Delphi 6 kann man damit zum Beispiel published Properties über ihren Namen ansprechen und Werte lesen/schreiben - Methoden mit Parametern werden erst mit Extended RTTI über Namen ansprechbar, das geht in Delphi 6 noch nicht.

Jumpy 21. Mär 2013 12:16

AW: Funktion über ihren Namen als String aufrufen
 
OK. Wäre denn folgender Workarround denkbar / sinnvoll:

Es gibt eine Funktion oder eine Klasse mit u.a. einer Funktion. Dieser wird der Name einer anderen Funktion als String übergeben und diese startet dann über ein "If...then...else usw."-Konstrukt die gewünschte Funktion.
Die Funktion muss natürlich alle diese Funktionen kennen und es wäre ggf. auch möglich Parameter zu übergeben (ist aber z.Zt. nicht wichtig).
Also so etwas wie eine Factory für Funktionen?

Zum Hintergrund: Bestimmte Funktionen sollen hintereinander ablaufen, die Ablaufreihenfolge soll aber veränderbar sein (über eine Tabelle gesteuert).

Sir Rufo 21. Mär 2013 12:28

AW: Funktion über ihren Namen als String aufrufen
 
Ja, warum denn nicht ...
Delphi-Quellcode:
procedure CallFunction( const AName : string; AParams : array of const; AResult : Variant );
var
  LName : string;
begin
  AResult := Null;
  LName := LowerCase( AName );
  if LName = 'tollefunktion' then
    tollefunktion
  else if LName = 'nochtollerefunktion' then
    nochtollerefunktion
  else
    raise Exception.CreateFmt( 'Funktion "%s" nicht gefunden!', [AName] );
end;

Jumpy 21. Mär 2013 12:33

AW: Funktion über ihren Namen als String aufrufen
 
Prima, danke für die Bestätigung.

Der schöne Günther 21. Mär 2013 12:59

AW: Funktion über ihren Namen als String aufrufen
 
Ich habe zwar das Delphi Language Coding Standards Document oder den Object Pascal Style Guide zwar noch nicht durchgeackert. Aber darf man trotzdem fragen, warum eigentlich das A als Prefix? :gruebel:

Bummi 21. Mär 2013 13:03

AW: Funktion über ihren Namen als String aufrufen
 
wenn Du den Link schon angibst:
Zitat:

The "A" prefix is a convention to disambiguate when the parameter name is the same as a property or field name in the class.

Sir Rufo 21. Mär 2013 13:12

AW: Funktion über ihren Namen als String aufrufen
 
Zitat:

Zitat von Der schöne Günther (Beitrag 1208207)
Ich habe zwar das Delphi Language Coding Standards Document oder den Object Pascal Style Guide zwar noch nicht durchgeackert. Aber darf man trotzdem fragen, warum eigentlich das A als Prefix? :gruebel:

Weil es dadurch klarer wird ;)
Delphi-Quellcode:
type
  TMyClass = class
  strict private
    _Value : string;
  private
    FValue : string;
    function GetValue : string;
  public
    function GetSomething( const AValue : string ) : string;
    property Value : string read GetValue;
  end;

function TMyClass.GetValue : string;
begin
  Result := '_' + FValue + '_';
end;

function TMyClass.GetSomething( const AValue : string ) : string;
var
  LValue : string;
begin
  LValue := 'Something';
 
  Result :=
    LValue + // local variable
    AValue + // attribute
    _Value + // strict private class field
    FValue + // private class field
    Value;  // property
end;

Der schöne Günther 21. Mär 2013 13:14

AW: Funktion über ihren Namen als String aufrufen
 
Puh, das wäre nicht mein Geschmack aber gut zu wissen :)

Andreas L. 21. Mär 2013 14:53

AW: Funktion über ihren Namen als String aufrufen
 
Wie man eine Funktion über ihren Namen als String aufruft ist mir nicht bekannt. Klassen lassen sich aber über den Namen erzeugen. Du könntest also deine Funktionen als Klassen abbilden und den auszuführenden Code im Constructor unterbringen. Einen Rückgabewert kannst du ggf. als Property definiern. Für eine ggf. nötige Parameter-Übergabe muss man sich noch was einfallen lassen, aber das geht bestimmt. Die Lösung ist aber ziemlich umständlich, vielleicht kann man dein Vorhaben mit einer Scripting-Engine besser umsetzen (evtl. JvInterpreter aus der JVCL).

Delphi-Quellcode:
uses
  Classes;

...

  { -------------------------------------------------------------------------- }
  { TCsObjectCreator --------------------------------------------------------- }
  { -------------------------------------------------------------------------- }

  TCsObjectCreator = class(TObject)
  public
    class function CreateObjectByClassName(AClassName: String): TObject;
    class function CreateObjectByClassType(AClassType: TClass): TObject;
    class function FindClassByName(AClassName: String): TClass;
  end;

{ ---------------------------------------------------------------------------- }
{ TCsObjectCreator ----------------------------------------------------------- }
{ ---------------------------------------------------------------------------- }

class function TCsObjectCreator.CreateObjectByClassName(AClassName: string): TObject;
var
  MetaClass: TPersistentClass;
begin
  MetaClass := TPersistentClass(Self.FindClassByName(AClassName));

  Result := MetaClass.Create;
end;

class function TCsObjectCreator.CreateObjectByClassType(AClassType: TClass): TObject;
begin
  Result := AClassType.Create;
end;

class function TCsObjectCreator.FindClassByName(AClassName: string): TClass;
begin
  Result := FindClass(AClassName);
end;
Die Klassen müssen im initialization-Abschnitt mit RegisterClass(es) registriert werden:

Unit mit den "Funktionen":
Delphi-Quellcode:
  TBaseFunction = class;

  TTestFunction = class;

  TBaseFunctionClass = class of TBaseFunction;

  TBaseFunction = class(TPersistent)
  protected
    FResult: String;
  public
    class function GetClassName: String; virtual;
  published
    property Result: String read FResult;
  end;

  TTestFunction = class(TBaseFunction)
  public
    constructor Create; virtual;
    class function GetClassName: String; override;
  end;

...

class function TBaseFunction.GetClassName: String;
begin
  Result := 'TBaseFunction';
end;

constructor TTestFunction.Create;
begin
  FResult := 'Hallo Welt';
  // irgendwas machen
end;

class function TTestFunction.GetClassName: String;
begin
  Result := 'TTestFunction';
end;

...

initialization
  RegisterClass(TTestFunction); // Edit: Evtl. muss der Parameter so aussehen: TPersistentClass(TTestFunction);
Aufruf aus dem Programm:
Delphi-Quellcode:
procedure TForm1.Button1Click(Sender: TObject);
var
  Func: TObject;
begin
  // Objekt mittels String aus Edit1 erzeugen, constructor wird automatisch ausgeführt
  Func := TCsObjectCreator.CreateObjectByClassName(Edit1.text); // in text steht z. B. TTestFunction
 
  // Objekt wurde erzeugt und ist abgeleitet von Basisklasse?
  if Assigned(Func) and (Func is TBaseFunction) then
  begin
    // Rückgabewert der "Funktion" anzeigen (oder in Var speichern)
    ShowMessage(TBaseFunction(Func).Result);
 
    // Erzeugtes Objekt freigeben
    FreeAndNil(Func);
  end;
end;
ungetestet

DeddyH 21. Mär 2013 15:02

AW: Funktion über ihren Namen als String aufrufen
 
Zumindest parameterlose published-Proceduren kann man so aufrufen:
Delphi-Quellcode:
procedure Call(AObject: TObject; const Methodname: string);
type
  TProc = procedure of object;
var
  Method: TMethod;
  Proc: TProc;
begin
  Method.Data := AObject;
  Method.Code := AObject.MethodAddress(Methodname);
  if Assigned(Method.Code) then
    begin
      Proc := TProc(Method);
      Proc;
    end;
end;
Leider weiß ich nicht, seit welcher Delphi-Version es TMethod gibt, in Delphi 2007 ist sie jedenfalls bereits vorhanden.

Andreas L. 21. Mär 2013 15:17

AW: Funktion über ihren Namen als String aufrufen
 
Zitat:

Zitat von DeddyH (Beitrag 1208239)
Zumindest parameterlose published-Proceduren kann man so aufrufen:
Delphi-Quellcode:
procedure Call(AObject: TObject; const Methodname: string);
type
  TProc = procedure of object;
var
  Method: TMethod;
  Proc: TProc;
begin
  Method.Data := AObject;
  Method.Code := AObject.MethodAddress(Methodname);
  if Assigned(Method.Code) then
    begin
      Proc := TProc(Method);
      Proc;
    end;
end;
Leider weiß ich nicht, seit welcher Delphi-Version es TMethod gibt, in Delphi 2007 ist sie jedenfalls bereits vorhanden.

Den Code (oder ähnlichen) habe ich mal in der Delphi-Hilfe gesehen. Auch nicht schlecht :) Rückgabewert und Parameter könnte man evtl. (unsauber) über globale Variablen über Felder bzw. Eigenschaften lösen. Der Vollständigkeit wegen:

Delphi-Quellcode:

  TForm1 = class(TForm)
    ...
  private
    FProcResult: String;
    FProcParams: Array of String;
  ...
  end;

...

//var
//  ProcResult: String; // oder Variant, ...
//  ProcParams: Array of String;

...

procedure TForm1.TestProc;
begin
  if FProcParams[Low(FProcParams)] = 'Hallo' then
    FProcResult := 'Hallo'
  else
    FProcResult := 'Welt';
end;

...

ButtonClick:
begin
  // parameter festlegen
  SetLength(FProcParams, 1);
  FProcParams[1] := 'Hallo';

  // Proc aufrufen
  Call(Self, 'TestProc');

  // Rückgabewert anzeigen
  ShowMessage(FProcResult);
end;

DeddyH 21. Mär 2013 15:23

AW: Funktion über ihren Namen als String aufrufen
 
Naja, wenn man aus Call eine (Klassen-)Methode macht, könnte man auf private (Klassen-)Felder zugreifen und somit auf globale Variablen verzichten.

Furtbichler 21. Mär 2013 15:31

AW: Funktion über ihren Namen als String aufrufen
 
Wenn man die Routinen in einer DLL vorhält, kann man diese zur Laufzeit unter ihrem Namen aufrufen.
Delphi-Quellcode:
Procedure CallProcedure (aDLLHandle : THandle; aFunctionName : String; aInParam : Variant; Var aOutParam : Variant);
Var
  pProcPtr: TDLLProcedure;

Begin
  pProcPtr := GetProcAddress(aDLLHandle, PChar(aFunctionName));
  If Not Assigned(pProcPtr) Then
    Raise Exception.CreateFmt('Function "%s" Not Found (%d)', [aFunctionName, GetLastError]);
  aOutParam := Null;
  pProcPtr(aInParam, aOutParam);
End;
Der Schnippsel ist aus einem Uralt-Projekt wo mit Plugins gearbeitet wurde.

Andreas L. 21. Mär 2013 15:51

AW: Funktion über ihren Namen als String aufrufen
 
Zitat:

Zitat von DeddyH (Beitrag 1208244)
Naja, wenn man aus Call eine (Klassen-)Methode macht, könnte man auf private (Klassen-)Felder zugreifen und somit auf globale Variablen verzichten.

Ist mir nach dem posten auch eingefallen. Hab' den Beitrag daher editiert.


Alle Zeitangaben in WEZ +1. Es ist jetzt 13:38 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 by Thomas Breitkreuz