![]() |
Delphi-Version: XE2
Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
Folgendes Szenario: Ich habe zwei Strings:
Ich möchte daraus ein TValue-Objekt machen. Natürlich mit dem Typ tkInteger. Es sind doch eigentlich nur zwei Schritte: Den zum String valueType passenden TTypeKind finden, und ein TValue-Objekt erstellen. Zu dem String den passenden Typ zu finden ist einfach:
Delphi-Quellcode:
.
valueType := TTypeKind(
GetEnumValue( TypeInfo(TTypeKind), valueTypeStr ) ); Nur wie lege ich nun ein TValue an? Alle generischen Funktionen wollen in ihren eckigen Klammern einen "expliziten Typen", etwas wie "Integer" oder "String". Den habe ich nicht. Ich habe doch nur meinen
Delphi-Quellcode:
:(.
TTypeKind
Bleibt mir nun nichts anderes übrig, als eine Riesen case-Anweisung aufzuziehen? "Wenn valueType = tkInteger dann meinTValue := TValue.From<Integer>(valueContent.toInteger());" usw. Mir gehen die Ideen aus :pale: |
AW: Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
Du könntest dir deinen eigenen Variant zusammenbasteln und einen Item, der ein Feld davon hat. Und dann eine Liste dieser Items. Dann bräuchtest du das case nur innerhalb dieses Items. Oder hab ich dich jetzt ganz falsch verstanden?
|
AW: Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
|
AW: Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
Ich bekomme es einfach nicht hin.
Mit einem Variant habe ich es auch mal versucht. Aber ich kann da ja nur meinen String reinstecken. Mit
Delphi-Quellcode:
komme ich auch nicht weiter: Ich kann problemlos ein TValue-Objekt mit dem Typ tkString erstellen. Aber ich bin absolut hilflos wenn ich gerne ein TValue-Objekt vom Typ
TValue.Make
Delphi-Quellcode:
hätte. Ich finde keine RTTI-Methode die auch nur den geringsten Versuch unternimmt, einen String nach Integer zu casten.
tkInteger
Natürlich kann ich das von Hand machen. Und das tue ich momentan auch für Integer, Single, Double, String, ... Aber das macht doch so keinen Spaß. Zusammenfassung: Ich möchte ein Feld oder eine Property eines Objekts über die RTTI setzen. Für
Delphi-Quellcode:
brauche ich ein
SetValue(..)
Delphi-Quellcode:
-Objekt. Ich kann nicht ein TValue-Objekt vom Typ String und dem Inhalt '42' nehmen. Ich bin auch zu dumm, ein TValue-Objekt vom Typ Integer anzulegen, obwohl ich weiß, dass es vom TTypeKind 'tkInteger' sein soll.
TValue
Was für ein schwarzer Tag. Anhang: ![]() |
AW: Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
Also die ganzen As... machen ja ein Typecast. Das geht nicht zwischen integer und string. Deshalb muß das händisch erfolgen:
Delphi-Quellcode:
s := '42';
// Die 3 Varianten sind möglich Value := TValue.From<string>(s); Value := TValue.Make(@s, TypeInfo(string), Value); Value := s; // Umwandlung, NICHT cast mit .AsInteger i := StrToIntDef(Value.AsString, 0); |
AW: Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
Zitat:
Um das Case und die entsprechenden Typenkonvertierungen der von dir zu unterstützenden TypeKinds kommst du nicht herum. Auch der Umweg über ein Variant wird dir nix bringen, denn auch ein Variant weiß intern, was für eine Art Typ drin steckt. Dieses wird beim TValue.FromVariant überprüft bzw übernommen. Schreibst du dort also einen Variant rein, in dem '42' steht (also ein String) wird dann ein TValue mit der TypeInfo string (TypeKind tkUString) daraus. Übergibst du das an SetValue einer Integer Property, knallts. Also brauchst du sowas:
Delphi-Quellcode:
Bei der Verwendung von TValue muss man sich immer in den Sinn rufen, dass diese keine Typenkonvertierung unterstützen, die über eine direkte Zuweisungskompatibilität hinausgeht (also z.b. keine Konvertierung eines strings in ein Integer, Datum in string, etc). Auch die As... Methoden nicht. Einzig und allein mit ToString kann das meiste als string rausrücken.
function FromString(const AValue: string; ATypeKind: TTypeKind): TValue;
begin case ATypeKind of tkInteger: Result := StrToInt(AValue); // nutzt die Implicit Operator Überladung, würde auch mit TValue.From<Integer>(StrToInt(AValue)) funktionieren end; end; |
AW: Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
Ganz genau so ist es :thumb:
Vielen Dank für die ausführliche Antwort! Find ich etwas blöd, dass es da nichts gibt. Ich habe intern auch etwas in der System.RTTI-Unit gewühlt. Da gibt es eine Matrix, welche Methoden bei Type-Casts von X nach Y aufgerufen werden. Bei String nach Integer versucht er es noch nichtmal, sondern nimmt die generische "Ne, das ist unmöglich"-Methode. Ich meine, man hätte es ja wenigstens versuchen können :-D Kein Beinbruch, aber blöd finde ich das trotzdem. Mit case über alle in Frage kommenden Typen zu rödeln ist hässlich. |
AW: Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
Funktionieren tut das schon:
Delphi-Quellcode:
type
TVarTypeEnum = (vtEmpty, vtNull, vtSmallint, vtInteger, vtSingle, vtDouble, vtCurrency, vtDate, vtOleStr, vtDispatch, vtError, vtBoolean, vtVariant, vtUnknown, vtDecimal, varUndef0F, vtShortInt, vtByte, vtWord, vtLongWord, vtInt64, vtUInt64); function GetRttiValueFromString(const AValue, AType: string): TValue; var ValueType: TVarTypeEnum; begin ValueType := TVarTypeEnum(GetEnumValue(TypeInfo(TVarTypeEnum), 'vtInteger')); Result := TValue.FromVariant(VarAsType('42', Integer(ValueType))); end; // benutzen: var MyValue: TValue; begin MyValue := GetRttiValueFromString('42', 'vtInteger'); ShowMessage('Inhalt: ' + MyValue.ToString + ' (' + MyValue.TypeInfo.Name + ')'); end; |
AW: Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
Zitat:
Zitat:
Btw ein case rödelt nicht, sondern ist ein direkter Sprung an die richtige Stelle ;) Ich finds übrigens eher hässlich, dass man irgendwo Werte als strings mit ihren TypeKinds speichert |
AW: Ich weiß den Typ und den Inhalt als String. Wie mache ich ein TValue daraus?
Zitat:
Wenn das jetzt ins Spiel gebrachte
Delphi-Quellcode:
(Danke an dieser Stelle) nun diese Lücke schließt, dann bin ich glücklich. Ändere ich die Gegenseite einfach ab, eben nicht den TTypeKind, sondern den TVarType als String zu schreiben. :twisted:
VarAsType
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 08:09 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