IEqualityComparer和字符串

时间:2018-10-12 06:40:12

标签: delphi delphi-xe7

为什么IEqualityComparer不使用字符串?

type
    TRec = record
        s: string;
    end;

var
    rec1, rec2: TRec;
    comparer: IEqualityComparer<TRec>;
    res: boolean;
begin
    rec1.s := 'a';
    rec2.s := 'a';

    comparer := TEqualityComparer<TRec>.default;
    res := comparer.equals(rec1, rec2);
    showMessage(boolToStr(res));

如果TRec条目包含数字值或一定长度的字符串,则IEqualityComparer可以正常工作。该代码如何工作?

1 个答案:

答案 0 :(得分:2)

编辑:

正如Rudy Velthuis在评论中注意到的那样,在新的Delphi版本中,相等的 constant 字符串的结果是正确的,因为它们共享相同的内存并具有相同的地址(所以我以前关于更好的RTTI的假设是错误的)

对于复杂类型,相等性默认比较器仍选择比较两个记录的原始字节的低级比较器-在我的示例的第二种情况下,相似字符串体的地址不同。

因此可靠的方法是构造自己的比较器以使用复杂类型-请参见下面的第三个示例。

type
    TRecS = record
        s: string;
    end;
 var
    rec1, rec2: TRecS;
    comparerS: IEqualityComparer<TRecS>;
    cmp: IEqualityComparer<TRecS>;
    res: boolean;
begin
    rec1.s := 'const';
    rec2.s := 'const';
    comparerS := TEqualityComparer<TRecS>.default;
    res := comparerS.equals(rec1, rec2);
    Memo1.Lines.Add(boolToStr(res));

    rec1.s := IntToStr(88);
    rec2.s := IntToStr(88);
    res := comparerS.equals(rec1, rec2);
    Memo1.Lines.Add(boolToStr(res));

    cmp := TEqualityComparer<TRecS>.Construct(
                      function(const Left, Right: TRecS): Boolean
                      begin
                          Result := Left.S = Right.S
                      end,
                      nil);
   res := cmp.equals(rec1, rec2);
   Memo1.Lines.Add(boolToStr(res));

    -1  //denotes true
    0  
    -1