Bug gefunden, Bug gefunden, Trallalla *drei-mal im Kreis hüpft*
Ich habe mal die
Unit etwas angepasst, da sie im Single Char modus gerne das letzte Zeichen geklaut hat.
Ich habe auch ctrl+shift+e benutzt um die etwas kurzen Name hoffentlich klarer zu machen.
Außerdem habe ich den Code mal in externe Iteratoren geworfen, für den Fall dass man die Ergebnisse a) nicht in einer Liste halten möchte und b) die
VCL nicht referenzieren will.
Gibt ja leider keinen String Container in der Delphi
RTL. (
RTL <>
VCL)
Und c) weil's so easy war
- vars zu feldern
- Init code im Constructor
- und einfachimmer raushüpfen wenn man was gefunden hat
Die Iteratoren (oder .Net speak Enumeratoren) sind aber eher Copy'nPaste + Anpassung, also keineswegs auf die Verwendung als Iterator optimiert...
Den Fehler und die Verwendung der Iteratoren kann man hiermit sehen:
Delphi-Quellcode:
uses
Classes,
csExplode2,
uExplodeEnumerators,
csExplode;
type
TOriginalStringDivider = csExplode.TStringDivider;
TStringDivider = csExplode2.TStringDivider;
procedure Original(const aPattern, aText : String);
var
s : string;
sl : TStringList;
sd : TOriginalStringDivider;
begin
sl := TStringList.Create();
sd := TOriginalStringDivider.Create();
sd.Explode(aPattern, aText, sl);
for s in sl do
Writeln(s);
sd.Free();
sl.Free();
end;
procedure UseSL(const aPattern, aText : String);
var
s : string;
sl : TStringList;
begin
sl := TStringList.Create();
TStringDivider.Explode(aPattern, aText, sl);
for s in sl do
Writeln(s);
sl.Free();
end;
procedure UseEnum(const aPattern, aText : String);
var
s : string;
begin
for s in TStringDivider.Explode(aPattern, aText) do
Writeln(s);
end;
procedure UseEnumDirectly(const aPattern, aText : String);
var
enum : IExplodeEnumerator;
begin
enum := TStringDivider.Explode(aPattern, aText) as IExplodeEnumerator;
while enum.MoveNext() do
Writeln(enum.Current);
end;
procedure RunAll(const aPattern, aText : String);
begin
Original(aPattern, aText);
Writeln('----------------------------');
UseSL(aPattern, aText);
Writeln('----------------------------');
UseEnum(aPattern, aText);
Writeln('----------------------------');
UseEnumDirectly(aPattern, aText);
Writeln('----------------------------');
end;
begin
ReportMemoryLeaksOnShutdown := true;
RunAll('abcxydefxyghixymmmxyx','y');
RunAll('abcxydefxyghixymmmxyy','y');
end.
btw: Warum hast du hier Instanzmethoden gewählt obwohl du gar keinen State zwichen den Calls halten musst?
Klassenmethoden hätten ja auch gereicht, bzw. sogar statische methoden in Delphi2006, wodurch du dir den impliziten parameter auf die class reference sparst:
Delphi-Quellcode:
type
TStringDivider = class
private
class procedure AddString(pStart, pEnd: PChar; aItems: TStrings); static;
class procedure QSExplode(const aText, aPattern: String; aItems: TStrings); static;
public
class procedure Explode(const aText, aPattern: String; aItems: TStrings); overload; static;
//class function Explode(const aText, aPattern: String) : IExplodeEnumerable; overload; static;
end;
btw: Wer in den unteren Procs (UseEnum*) nach Free sucht, sucht vergebens, da ich mit Interfaces arbeite überlasse ich das dem Compiler und der Referenzzählung.