![]() |
Re: verschiedene Sets an Prozedur übergeben
Zitat:
Das was du uns jetzt zeigst ändert die Sache erheblich, allerdings ist eben meine Glaskugel zum Würfel eingeschmolzen worden, sorry das ich also nicht mehr hellsehen kann. Mache für jedes deiner Sets eine überladene Funktion. Innerhalb dieser Funktion castest du hart deine übergebenen Sets in einen Integer und rufst eine interne Funktion auf die mit einem Integer als Param arbeutet, also so
Delphi-Quellcode:
Was daran ist nun so aufwendig, ich habe 30 Sekunden benötigt.
procedure SaveSet(const Set: TMySet1); overload;
procedure SaveSet(const Set: TMySet2); overload; procedure SaveSet(const Set: Integer); overload; procedure SaveSet(const Set: TMySet1); begin SaveSet(Integer(Set)); end; procedure SaveSet(const Set: TMySet2); begin SaveSet(Integer(Set)); end; procedure SaveSet(const Set: Integer); begin IniFile.WriteInteger(Set.....); end; Gruß Hagen |
Re: verschiedene Sets an Prozedur übergeben
Hagen iss au en schöner Vorschlag, ich habs jetzt aber so:
Delphi-Quellcode:
Hier kann ich jedes meiner Sets übergeben.
type
TConvClass = (c1,c2,c3,c4,c5,c6,c7,c8); TConvSet = set of TConvClass; function TForm1.Test(const ASet):Integer; var x : TConvClass; begin Result:=0; for x in TConvSet(ASet) do begin if Ord(x) = 0 then Result:=Result+1 else Result:=Result+Round(Power(Ord(x),2)); end; end; bei Set of Byte funktioniert das nicht, da scheint alles drin zu sein und ich bekomm anstatt 15 irgendwas wie 20937223 oder so.. also nicht geeignet. Aber jetzt gehts ja. |
Re: verschiedene Sets an Prozedur übergeben
Delphi-Quellcode:
Also diesen Code könntest du mir mal erklären, also nicht was er macht sondern warum du es so machst ?
function TForm1.Test(const ASet):Integer;
var x : TConvClass; begin Result:=0; for x in TConvSet(ASet) do begin if Ord(x) = 0 then Result:=Result+1 else Result:=Result+Round(Power(Ord(x),2)); end; end;
Delphi-Quellcode:
sollte nämlich exakt das Gleiche machen.
function TForm1.Test(const ASet):Integer;
begin Result := Ord(TConvSet(ASet)); end; Gruß Hagen |
Re: verschiedene Sets an Prozedur übergeben
Zitat:
Sagen wir ich bin relativer bis absoluter Programmier-Anfänger und bin eninfach nicht drauf gekommen :duck: Allerdings hast du en kleinen fehler, bei TConvSet bringt er den Fehler ungülter Typ, wenn ich Byte nehm gehts wieder. Also ich hab jetzt die folgenden beiden Funktionen:
Delphi-Quellcode:
geht die IntToSet Funktion auch einfacher oder ist das OK?function SetToInt(const ASet):Integer; begin Result:=Ord(Byte(ASet)); end; Und zum zurückwandeln in ein Set: function IntToSet(AInt : Integer):TConvSet; var x : Integer; LInt : Integer; begin Result:=[]; LInt := AInt; for x := 32 downto 0 do if LInt > 0 then begin if Round(Power(2,x)) <= LInt then begin LInt:=LInt-Round(Power(2,x)); Result:=Result + [TConvClass(x)]; end else if (LInt = 1) then begin LInt:=LInt-1; Result:=Result + [TConvClass(0)]; exit; end; end; end; aufgerufen wird das ganze so: procedure Test; var testset : TMySet; setint : Integer; begin testset:=TMySet(IntToSet(5)); setint:=SetToint(testset); end; |
Re: verschiedene Sets an Prozedur übergeben
Zitat:
Sagen wir ich bin relativer bis absoluter Programmier-Anfänger und bin eninfach nicht drauf gekommen :duck: Allerdings hast du en kleinen fehler, bei TConvSet bringt er den Fehler ungülter Typ, wenn ich Byte nehm gehts wieder. Also ich hab jetzt die folgenden beiden Funktionen:
Delphi-Quellcode:
geht die IntToSet Funktion auch einfacher oder ist das OK?function SetToInt(const ASet):Integer; begin Result:=Ord(Byte(ASet)); end; Und zum zurückwandeln in ein Set: function IntToSet(AInt : Integer):TConvSet; var x : Integer; LInt : Integer; begin Result:=[]; LInt := AInt; for x := 32 downto 0 do if LInt > 0 then begin if Round(Power(2,x)) <= LInt then begin LInt:=LInt-Round(Power(2,x)); Result:=Result + [TConvClass(x)]; end else if (LInt = 1) then begin LInt:=LInt-1; Result:=Result + [TConvClass(0)]; exit; end; end; end; aufgerufen wird das ganze so: procedure Test; var testset : TMySet; setint : Integer; begin testset:=TMySet(IntToSet(5)); setint:=SetToint(testset); end; |
Re: verschiedene Sets an Prozedur übergeben
Delphi-Quellcode:
Allerdings meine ich das du besser deine Set's als String abspeicherst. Du machst das doch um das in einer Config zu speichern, oder ? Dort wären ja lesbare Strings viel besser, also sowas in der Art
function IntToSet(Value: INteger): TConvSet;
begin Result := TConvSet(Byte(Vale)); end; '[c2,c5,c6]' Du hast nämlich in Delphi über die RTTI und der Unit TypInfo.pas die Möglichkeit deine TConvSet Variable in einen String und zurück zu konvertieren. Dabei enthält dieser String exakt die Elementnamen deines Sets die du in deinem Quelltext als Deklaration verwendet hast. Gruß Hagen |
Re: verschiedene Sets an Prozedur übergeben
Mit Strings könnte das dann so aussehen
Delphi-Quellcode:
Benutzt wird es dann so
unit MyUnit;
uses .....; interface type TA = (ta1, ta2, ta3, ta4, ta5); TSetA = set of TA; TB = (tb1, tb2, tb3, tb4, tb5); TSetB = set of TB; function GetAsString(Value: TSetA): String; overload; function GetAsString(Value: TSetB): String; overload; procedure SetAsString(var Data: TSetA; const Value: String); overload; procedure SetAsString(var Data: TSetB; const Value: String); overload; implementation uses SysUtils, TypInfo, .....; procedure CheckType(Info: PTypeData); begin Assert(Info.MaxValue - Info.MinValue <= 32, 'Set ist zu groß'); end; function GetAsString(Data: Cardinal; Info: PTypeInfo): String; overload; var T: PTypeInfo; P: PTypeData; I: LongInt; begin T := GetTypeData(Info).CompType^; P := GetTypeData(T); CheckType(P); for I := P.MinValue to P.MaxValue do if Data and (1 shl I) <> 0 then Result := Result + GetEnumName(T, I) + ','; SetLength(Result, Length(Result) -1); end; function SetAsString(Value: String; Info: PTypeInfo): Cardinal; overload; function NextWord(var C: PChar): String; const cSep = [',', ';', ' ', '[', ']', '{', '}', '(', ')']; var I: Integer; begin while C^ in cSep do Inc(C); I := 0; while not (C[I] in cSep + [#0]) do Inc(I); SetString(Result, C, I); Inc(C, I); end; var T: PTypeInfo; N: String; V: Integer; C: PChar; begin Result := 0; T := GetTypeData(Info).CompType^; CheckType(GetTypeData(T)); C := PChar(Value); while True do begin N := NextWord(C); if N = '' then Break; V := GetEnumValue(T, N); if V < 0 then raise Exception.CreateFmt('Ungültes Element "%s" im Set', [N]); Result := Result or (1 shl V); end; end; function GetAsString(Value: TSetA): String; overload; begin Result := GetAsString(Byte(Value), TypeInfo(TSetA)); end; function GetAsString(Value: TSetB): String; overload; begin Result := GetAsString(Byte(Value), TypeInfo(TSetB)); end; procedure SetAsString(var Data: TSetA; const Value: String); overload; begin Byte(Data) := SetAsString(Value, TypeInfo(TSetA)); end; procedure SetAsString(var Data: TSetB; const Value: String); overload; begin Byte(Data) := SetAsString(Value, TypeInfo(TSetB)); end; end.
Delphi-Quellcode:
Ich empfehle dir mit überladenen Prozeduren zu arbeiten und so wie oben die Zugriffe lokal zu kapseln. Das führt dann dazu das diese Unit als Schnittstelle in deinem Program absolut sauber ist, ohne das man extern noch unsauber casten muß.uses MyUnit; var A: TSetA; B: TSetB; begin SetAsString(A, '[ta1,ta3,ta5]'); SetAsString(B, 'tb2,tb4'); ShowMessage( GetAsString(A) ); ShowMessage( GetAsString(B) ); end; Je nach Größe deiner Mengen musst du intern entsprechend casten. Es gibt 3 ordinale Typen von Sets in Delphi. Set mit <= 8 Elementen müssen nach Byte casted werden, Sets mit <= 16 Elementen nach Word und Sets mit <= 32 nach Cardinal. Größere Mengen werden im obigen Source nicht unterstützt !! Gruß Hagen |
Re: verschiedene Sets an Prozedur übergeben
nehee.. moooment :)
Also: So isses jetzt geplant: Set -> Integer Integer -> an php-Skript schicken in php-Skript evtl. verarbeiten und in DB speichern später aus DB auslesen, wieder in php-Skript verarbeiten(Schadenstypen für Kämpfe auslesen) Zahl von Skript zurück ans Programm, dort wieder in Set da finde ich persönlich das ganze als Zahl schon vorteilhafter |
Re: verschiedene Sets an Prozedur übergeben
Jo ist ja auch kein Problem ;)
Gruß Hagen |
Alle Zeitangaben in WEZ +1. Es ist jetzt 04:45 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