![]() |
Delphi 2007 - UpperCase(), LowerCase() Implementierung
Hallo,
Kann mir jemand, der die Vollversion von Delphi 2007 hat, sagen, wie die Funktionen UpperCase() und LowerCase() dort intern implementiert sind? (Ich hoffe, dass ist erlaubt.) Ich habe leider nur die Testversion. Ich frage das, weil mich erstens interessiert, wieso diese Funktionen so extrem schnell sind - keine 2 Sekunden für 100.000.000 Durchläufe! Selbst eine Funktion, mit gerade mal einer einzigen Zuweisung, ist fast 3x so langsam! Wie kann das sein? :gruebel: Optimierte UpperCase() Funktionen, wie z.B ![]() Und zweitens brauche ich genau diese schnellen Funktionen als Prozedur mit Referenzparameter, also so, dass der Eingabestring direkt verändert wird. (Diese würde ich mir dann selber bauen, wenn ich den Quellcode hätte.) Ich hoffe auf Hilfe, Danke! :) Hatte sogar eine Null vergessen^^ |
Re: Delphi 2007 - UpperCase(), LowerCase() Implementierung
Lade Dir die FastCode Library runter, da ist die UpperCase Variante (fast) die gleiche.
Interessant wäre auch, wie Du die Funktionen testest. Nicht das Du einer Optimierung in die Falle gegangen bist. |
Re: Delphi 2007 - UpperCase(), LowerCase() Implementierung
ich glaub da bist du eher einer Speicheroptimierung zum Opfer gefallen
UpperCase macht nicht viel, außer den Eingabestring zu kopieren und dabei bei allen Zeichen von "a" bis "z" um Ord(32) anzuheben die Stringrückgabewerde in Delphi werden intern als Var-Parameter übergeben also
Delphi-Quellcode:
wird in das hier übersetzt
function UpperCase(const S: String): String;
begin ... end; for i := 1 to 1000 do UpperCase(S);
Delphi-Quellcode:
und wie du nun bemerkst, wird hier kaum etwas gemacht, außer die Zeichen zu kopieren
procedure UpperCase(const S: String Var Result: String);
var i: Integer; C: Char; begin SetLength(Result, Length(S)); for i := 1 to Length(S) do begin C := S[i]; if C in ['a'..'z'] then Inc(C, 32); Result[i] := C; end; end; for i := 1 to 1000 do UpperCase(S, TempString); der Flaschenhals bei Stringoperationen ist aber das reservieren/freigeben des Strings und dieses wird hier nur je einmal am Anfang und am Ende gemacht und ansonsten ist der String selber ja komplett unverändert. PS: das UpperCase da oben ist die unoptimierte Kurzfassung dessen, was die Funktion macht Und FastUpcase ruft intern je Zeichen auch noch eine Extrafunktion (UpCase) auf, wärend UpperCase das selber macht (pro Zeichen 2 Sprungoperationen weniger) ein Test mit AnsiUpperCase würde in Delphi wohl auch andere Ergebnisse liefern, da dieses etwas genauer übersetzt (samt umlauten wie äöü) oder versuche mal diese Version
Delphi-Quellcode:
in der anderen Version (siehe dein Link)
function FastUpperCase(const S: String): String;
var C: ^Char; I: Integer; begin SetLength(Result, Length(S); C := Addr(Result[1]); for I := 0 to Length(Result) - 1 do begin C^ := UpCase(C^); C := Ptr(Integer(C) + SizeOf(Char)); end; end; wird eine Kopie von S angelegt, das die eigentliche Result-Variable freigegeben und durch die Kopie ersetzt (also Speicher freigeben+Speicher reservieren) SetLength sorgt aber auch für einen "unique" String, allerdings ändert (oder auch nicht, wenn er es schon ist) es hier nur den Result und ersetzt ihn nicht |
Re: Delphi 2007 - UpperCase(), LowerCase() Implementierung
@Satty: Danke, das sieht schon ganz gut aus, nur leider kann ich davon nichts ändern, weil ich von Assembler keine Ahnung habe. :o
Diese ganzen FastCode Sachen sind auch etwas langsamer (~500ms) Getestet habe ich das übrigens so:
Delphi-Quellcode:
(Ich weiss, GetTickCount() ist sehr ungenau, aber hier sollte es doch reichen?)
var a:cardinal;i:integer;s:string;
begin a:=gettickcount; for i := 1 to 100000000 do s:=uppercase('hallo'); showmessage(inttostr(gettickcount-a)); end; Also Eigentlich suche ich nur eine Funktion, die genauso oder schneller ist und so arbeitet:
Delphi-Quellcode:
procedure UpperCase(var S: string);
@himitsu Danke, ich habe es jetzt mal so gemacht:
Delphi-Quellcode:
Weit unter 2 Sekunden :thumb: Damit kann ich leben.
procedure FastUpperCase(var S: String); //inline;
var C: ^Char; I,j: Integer; begin C := Addr(S[1]); j:=sizeof(char); for I := 0 to Length(S) - 1 do begin if (C^ >= 'a') and (C^ <= 'z') then C^ := UpCase(C^); C := Ptr(Integer(C) + j); end; end; |
Re: Delphi 2007 - UpperCase(), LowerCase() Implementierung
dieser Test entspricht aber keinesfalls den realen Bedingungen, also im Grunde kannst du diese Testergebnisse vergessen, da sich ja wohl im Normalfalls der Eingangsstring unterscheidet.
um wenigstens ein bissl den Realbedingungen nahzukommen, versuch mal die Funktionen hiermit zu testen:
Delphi-Quellcode:
var a:cardinal;i:integer;s,s1,s2:string;
a:=gettickcount; s1:='hallo'; s2:='hallo2'; for i := 1 to 10000000 do begin if Odd(i) then s := s1 else s := s2; s := uppercase(s); if s = '' then ; end; showmessage(inttostr(gettickcount-a)); |
Re: Delphi 2007 - UpperCase(), LowerCase() Implementierung
Zitat:
Delphi-Quellcode:
procedure FastUpperCase(var S: String);
var P: PChar; Len: Integer; begin { Garantieren, dass keine andere String-Referenz auf unseren Speicher zeigt (Copy-on-Write) } UniqueString(S); {$STRINGCHECKS OFF} // grrr... Wir wollen diesen Unsinn nicht, der den Code Len := Length(S); // nur aufbläht, verlangsamt und für Delphi und gescheite {$STRINGCHECKS ON} // C++Builder Entwickler keinen Mehrwehrt bringt. P := PChar(S); for Len := Len downto 1 do begin case P^ of // Delphi 2009 Workaround für "in ['a'..'z']" Warnung, gleicher Maschinencode 'a'..'z': P^ := Char(Ord(P^) xor $20); // Die Eigenschaften der ASCII-Tabelle ausnutzen end; Inc(P); end; end; |
Re: Delphi 2007 - UpperCase(), LowerCase() Implementierung
Delphi-Quellcode:
Das existiert bei mit gar nicht:
{$STRINGCHECKS OFF} // grrr... Wir wollen diesen Unsinn nicht, der den Code
-> [DCC Fehler] mlUtils.pas(6165): E1030 Ungültige Compileranweisung: 'STRINGCHECKS' Hier aber nochmal die Deutsche Version (ö,ä,ü) + LowerCase:
Delphi-Quellcode:
procedure UpCaseFst(var S: string); overload;
var P: PChar; Len: Integer; begin { Garantieren, dass keine andere String-Referenz auf unseren Speicher zeigt (Copy-on-Write) } UniqueString(S); Len := Length(S); // nur aufbläht, verlangsamt und für Delphi und gescheite P := PChar(S); for Len := Len downto 1 do begin case P^ of // Delphi 2009 Workaround für "in ['a'..'z']" Warnung, gleicher Maschinencode 'a'..'z': P^ := Char(Ord(P^) xor $20); // Die Eigenschaften der ASCII-Tabelle ausnutzen 'ö': P^ := 'Ö'; 'ä': P^ := 'Ä'; 'ü': P^ := 'Ü'; end; Inc(P); end; end;
Delphi-Quellcode:
Ob das jetzt immernoch die Highendoptimierung ist, weiß ich nicht. :spin2:
procedure LwCaseFst(var S: string); overload;
var P: PChar; Len: Integer; begin { Garantieren, dass keine andere String-Referenz auf unseren Speicher zeigt (Copy-on-Write) } UniqueString(S); Len := Length(S); // nur aufbläht, verlangsamt und für Delphi und gescheite P := PChar(S); for Len := Len downto 1 do begin case P^ of // Delphi 2009 Workaround für "in ['a'..'z']" Warnung, gleicher Maschinencode 'A'..'Z': P^ := Char(Ord(P^) or $20); // Die Eigenschaften der ASCII-Tabelle ausnutzen 'Ö': P^ := 'ö'; 'Ä': P^ := 'ä'; 'Ü': P^ := 'ü'; end; Inc(P); end; end; |
Re: Delphi 2007 - UpperCase(), LowerCase() Implementierung
Zitat:
|
Re: Delphi 2007 - UpperCase(), LowerCase() Implementierung
Zitat:
|
Re: Delphi 2007 - UpperCase(), LowerCase() Implementierung
Zitat:
Delphi lässt öäü einfach wie es ist. Nur dieses Verhalten wurd mir gestern heute morgen zum Verhängnis und beim Ändern hab ich nicht drauf geachtet wie Delphi das handhabt. Eigentlich wollte ich hier nur die LowCase Variante mit dem or-Operator darstellen. Wer öäü immer selber im Programm behandeln will, kann das ja schnell rausnehmen. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 20:44 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