How can I search faster for name/value pairs in a Delphi TStringList?

前端 未结 5 1578

I implemented language translation in an application by putting all strings at runtime in a TStringList with:

procedure PopulateStringList;
begin  
  EnglishStri         


        
相关标签:
5条回答
  • 2021-02-04 18:46

    If THashedStringList works for you, that's great. Its biggest weakness is that every time you change the contents of the list, the Hash table is rebuilt. So it will work for you as long as your list remains small or doesn't change very often.

    For more info on this, see: THashedStringList weakness, which gives a few alternatives.

    If you have a big list that may be updated, you might want to try GpStringHash by gabr, that doesn't have to recompute the whole table at every change.

    0 讨论(0)
  • 2021-02-04 18:51

    You can also use a CLASS HELPER to re-program the "IndexOfName" function:

    TYPE
      TStringsHelper = CLASS HELPER FOR TStrings
                         FUNCTION IndexOfName(CONST Name : STRING) : INTEGER;
                       END;
    
    FUNCTION TStringsHelper.IndexOfName(CONST Name : STRING) : INTEGER;
      VAR
        SL  : TStringList ABSOLUTE Self;
        S,T : STRING;
        I   : INTEGER;
    
      BEGIN
        IF (Self IS TStringList) AND SL.Sorted THEN BEGIN
          S:=Name+NameValueSeparator;
          IF SL.Find(S,I) THEN
            Result:=I
          ELSE IF (I<0) OR (I>=Count) THEN
            Result:=-1
          ELSE BEGIN
            T:=SL[I];
            IF CompareStrings(COPY(T,1,LENGTH(S)),S)=0 THEN Result:=I ELSE Result:=-1
          END;
          EXIT
        END;
        Result:=INHERITED IndexOfName(Name)
      END;
    

    (or implement it in a descendant TStrings class if you dislike CLASS HELPERs or don't have them in your Delphi version).

    This will use a binary search on a sorted TStringList and a sequential search on other TStrings classes.

    0 讨论(0)
  • 2021-02-04 18:54

    I think that you don't use the EnglishStringList(TStringList) correctly. This is a sorted list, you add elements (strings), you sort it, but when you search, you do this by a partial string (only the name, with IndexOfName).

    If you use IndexOfName in a sorted list, the TStringList can't use Dicotomic search. It use sequential search.

    (this is the implementation of IndexOfName)

      for Result := 0 to GetCount - 1 do
      begin
        S := Get(Result);
        P := AnsiPos('=', S);
        if (P <> 0) and (CompareStrings(Copy(S, 1, P - 1), Name) = 0) then Exit;
      end;
    

    I think that this is the reason of poor performance.
    The alternative is use 2 TStringList:
    * The first (sorted) only containts the "Name" and a pointer to the second list that contain the value; You can implement this pointer to the second list using the "pointer" of Object property.
    * The second (not sorted) list containt the values.

    When you search, you do it at first list; In this case you can use the Find method. when you find the name, the pointer (implemented with Object property) give you the position on second list with the value.

    In this case, Find method on Sorted List is more efficient that HashList (that must execute a funcion to get the position of a value).

    Regards.

    Pd:Excuse-me for mistakes with english.

    0 讨论(0)
  • 2021-02-04 18:58

    THashedStringList should be better, I think.

    0 讨论(0)
  • 2021-02-04 19:00

    In Delphi 2009 or later I would use TDictionary< string,string > in Generics.Collections. Also note that there are free tools such as http://dxgettext.po.dk/ for translating applications.

    0 讨论(0)
提交回复
热议问题