3

我有一个看起来像这样的记录:

TBigint = record
    PtrDigits: Pointer;                  <-- The data is somewhere else.
    Size: Byte;
    MSB: Byte;
    Sign: Shortint;
    ...
    class operator Implicit(a: TBigint): TBigint;  <<-- is this allowed?
    ....

该代码是前类运算符遗留代码,但我想添加运算符。

我知道数据确实应该存储在一个动态的字节数组中,但我不想更改代码,因为所有内容都在 x86-assembly 中。

我想按照代码来触发底部的类运算符:

procedure test(a: TBignum);
var b: TBignum;
begin
  b:= a;  <<-- naive copy will tangle up the `PtrDigit` pointers.
  ....

如果我将隐式类型转换添加到自身,是否会执行以下代码?

class operator TBigint.Implicit(a: TBigint): TBigint;
begin
  sdpBigint.CreateBigint(Result, a.Size);
  sdpBigint.CopyBigint(a, Result);
end;

(如果它按我的预期工作,将测试并添加答案)。

4

3 回答 3

4

我的第一个答案试图劝阻覆盖赋值运算符的想法。我仍然支持这个答案,因为要遇到的许多问题都可以通过对象更好地解决。

然而,David 非常正确地指出,它TBigInt是作为记录来实现的,以利用运算符重载。即a := b + c;。这是坚持基于记录的实现的一个很好的理由。

因此,我提出了一种用一块石头杀死两只鸟的替代解决方案:

  • 它消除了我在其他答案中解释的内存管理风险。
  • 并提供了一种简单的机制来实现 Copy-on-Write 语义。

我仍然建议,除非有充分的理由保留基于记录的解决方案,否则请考虑切换到基于对象的解决方案。

大体思路如下:

  • 定义一个接口来表示 BigInt 数据。(这最初可以是极简的并且仅支持指针的控制 - 就像在我的示例中一样。这将使现有代码的初始转换更容易。)
  • 定义将由TBigInt记录使用的上述接口的实现。
  • 接口解决了第一个问题,因为接口是托管类型;当记录超出范围时,Delphi 将取消对接口的引用。因此,底层对象将在不再需要时自行销毁。
  • 该接口还提供了解决第二个问题的机会,因为我们可以通过检查RefCount来知道我们是否应该 Copy-On-Write。
  • 请注意,从长远来看,将一些 BigInt 实现从记录移动到类和接口可能会证明是有益的。

以下代码是精简的“big int”实现,纯粹是为了说明这些概念。(即“大”整数仅限于常规的 32 位数字,并且只实现了加法。)

type
  IBigInt = interface
    ['{1628BA6F-FA21-41B5-81C7-71C336B80A6B}']
    function GetData: Pointer;
    function GetSize: Integer;
    procedure Realloc(ASize: Integer);
    function RefCount: Integer;
  end;

type
  TBigIntImpl = class(TInterfacedObject, IBigInt)
  private
    FData: Pointer;
    FSize: Integer;
  protected
    {IBigInt}
    function GetData: Pointer;
    function GetSize: Integer;
    procedure Realloc(ASize: Integer);
    function RefCount: Integer;
  public
    constructor CreateCopy(ASource: IBigInt);
    destructor Destroy; override;
  end;

type
  TBigInt = record
    PtrDigits: IBigInt;
    constructor CreateFromInt(AValue: Integer);
    class operator Implicit(AValue: TBigInt): Integer;
    class operator Add(AValue1, AValue2: TBigInt): TBigInt;
    procedure Add(AValue: Integer);
  strict private
    procedure CopyOnWriteSharedData;
  end;

{ TBigIntImpl }

constructor TBigIntImpl.CreateCopy(ASource: IBigInt);
begin
  Realloc(ASource.GetSize);
  Move(ASource.GetData^, FData^, FSize);
end;

destructor TBigIntImpl.Destroy;
begin
  FreeMem(FData);
  inherited;
end;

function TBigIntImpl.GetData: Pointer;
begin
  Result := FData;
end;

function TBigIntImpl.GetSize: Integer;
begin
  Result := FSize;
end;

procedure TBigIntImpl.Realloc(ASize: Integer);
begin
  ReallocMem(FData, ASize);
  FSize := ASize;
end;

function TBigIntImpl.RefCount: Integer;
begin
  Result := FRefCount;
end;

{ TBigInt }

class operator TBigInt.Add(AValue1, AValue2: TBigInt): TBigInt;
var
  LSum: Integer;
begin
  LSum := Integer(AValue1) + Integer(AValue2);
  Result.CreateFromInt(LSum);
end;

procedure TBigInt.Add(AValue: Integer);
begin
  CopyOnWriteSharedData;

  PInteger(PtrDigits.GetData)^ := PInteger(PtrDigits.GetData)^ + AValue;
end;

procedure TBigInt.CopyOnWriteSharedData;
begin
  if PtrDigits.RefCount > 1 then
  begin
    PtrDigits := TBigIntImpl.CreateCopy(PtrDigits);
  end;
end;

constructor TBigInt.CreateFromInt(AValue: Integer);
begin
  PtrDigits := TBigIntImpl.Create;
  PtrDigits.Realloc(SizeOf(Integer));
  PInteger(PtrDigits.GetData)^ := AValue;
end;

class operator TBigInt.Implicit(AValue: TBigInt): Integer;
begin
  Result := PInteger(AValue.PtrDigits.GetData)^;
end;

以下测试是在我构建建议的解决方案时编写的。他们证明:一些基本功能,写时复制按预期工作,并且没有内存泄漏。

procedure TTestCopyOnWrite.TestCreateFromInt;
var
  LBigInt: TBigInt;
begin
  LBigInt.CreateFromInt(123);
  CheckEquals(123, LBigInt);
  //Dispose(PInteger(LBigInt.PtrDigits)); //I only needed this until I 
                                          //started using the interface
end;

procedure TTestCopyOnWrite.TestAssignment;
var
  LValue1: TBigInt;
  LValue2: TBigInt;
begin
  LValue1.CreateFromInt(123);
  LValue2 := LValue1;
  CheckEquals(123, LValue2);
end;

procedure TTestCopyOnWrite.TestAddMethod;
var
  LValue1: TBigInt;
begin
  LValue1.CreateFromInt(123);
  LValue1.Add(111);

  CheckEquals(234, LValue1);
end;

procedure TTestCopyOnWrite.TestOperatorAdd;
var
  LValue1: TBigInt;
  LValue2: TBigInt;
  LActualResult: TBigInt;
begin
  LValue1.CreateFromInt(123);
  LValue2.CreateFromInt(111);

  LActualResult := LValue1 + LValue2;

  CheckEquals(234, LActualResult);
end;

procedure TTestCopyOnWrite.TestCopyOnWrite;
var
  LValue1: TBigInt;
  LValue2: TBigInt;
begin
  LValue1.CreateFromInt(123);
  LValue2 := LValue1;

  LValue1.Add(111); { If CopyOnWrite, then LValue2 should not change }

  CheckEquals(234, LValue1);
  CheckEquals(123, LValue2);
end;

编辑

添加了一个测试,演示TBigInt如何在过程中使用 as value 参数。

procedure TTestCopyOnWrite.TestValueParameter;
  procedure CheckValueParameter(ABigInt: TBigInt);
  begin
    CheckEquals(2, ABigInt.PtrDigits.RefCount);
    CheckEquals(123, ABigInt);
    ABigInt.Add(111);
    CheckEquals(234, ABigInt);
    CheckEquals(1, ABigInt.PtrDigits.RefCount);
  end;
var
  LValue: TBigInt;
begin
  LValue.CreateFromInt(123);
  CheckValueParameter(LValue);
end;
于 2013-09-15T19:30:07.193 回答
2

Delphi 中没有任何东西可以让您进入分配过程。Delphi 没有 C++ 复制构造函数。

您的要求是:

  1. 您需要对数据的引用,因为它是可变长度的。
  2. 您还需要值语义。

满足这两个要求的唯一类型是本机 Delphi 字符串类型。它们被实现为参考。但是他们的写时复制行为赋予了他们价值语义。由于您想要一个字节数组,因此 AnsiString 是满足您需求的字符串类型。

另一种选择是简单地使您的类型不可变。这将使您不必担心复制引用,因为引用的数据永远不会被修改。

于 2013-09-15T08:24:44.590 回答
1

在我看来,你TBigInt应该是一门课而不是记录。因为您担心 PtrDigits 被缠结,所以听起来您需要对指针引用的内容进行额外的内存管理。由于记录不支持析构函数,因此无法自动管理该内存。此外,如果您仅声明 的变量TBigInt,但不调用CreatBigInt构造函数,则内存未正确初始化。同样,这是因为您无法覆盖记录的默认无参数构造函数。

基本上,您必须始终记住已为记录分配的内容并记住手动解除分配。当然,您可以在记录中有一个取消分配程序来帮助解决这方面的问题,但您仍然必须记住在正确的位置调用它。

尽管如此,您可以实现一个显式Copy函数,并将一个TBitInt已正确复制的项目添加到您的代码审查清单中。不幸的是,您必须非常小心隐含的副本,例如通过值参数将记录传递给另一个例程。

以下代码说明了一个概念上与您的需求相似的示例,并演示了CreateCopy函数如何“解开”指针。它还强调了一些突然出现的内存管理问题,这就是为什么记录可能不是一个好的方法。

type
  TMyRec = record
    A: PInteger;
    function CreateCopy: TMyRec;
  end;

function TMyRec.CreateCopy: TMyRec;
begin
  New(Result.A);
  Result.A^ := A^;
end;

var
  R1, R2: TMyRec;
begin
  New(R1.A); { I have to manually allocate memory for the pointer 
               before I can use the reocrd properly.
               Even if I implement a record constructor to assist, I
               still have to remember to call it. }
  R1.A^ := 1;
  R2 := R1;
  R2.A^ := 2; //also changes R1.A^ because pointer is the same (or "tangled")
  Writeln(R1.A^);

  R2 := R1.CreateCopy;
  R2.A^ := 3; //Now R1.A is different pointer so R1.A^ is unchanged
  Writeln(R1.A^);
  Dispose(R1.A);
  Dispose(R2.A); { <-- Note that I have to remember to Dispose the additional 
                   pointer that was allocated in CreateCopy }
end;

简而言之,您似乎正试图用大锤来做他们并不真正适合做的事情。
他们非常擅长制作精确的副本。它们具有简单的内存管理:声明一个记录变量,并分配所有内存。变量超出范围,所有内存都被释放。


编辑

重写赋值运算符如何导致内存泄漏的示例。

var
  LBigInt: TBigInt;
begin
  LBigInt.SetValue(123);
  WriteBigInt(LBigInt); { Passing the value by reference or by value depends
                          on how WriteBigInt is declared. }
end;

procedure WriteBigInt(ABigInt: TBigInt);
//ABigInt is a value parameter.
//This means it will be copied.
//It must use the overridden assignment operator, 
//  otherwise the point of the override is defeated.
begin
  Writeln('The value is: ', ABigInt.ToString);
end;
//If the assignment overload allocated memory, this is the only place where an
//appropriate reference exists to deallocate.
//However, the very last thing you want to do is have method like this calling 
//a cleanup routine to deallocate the memory....
//Not only would this litter your code with extra calls to accommodate a 
//problematic design, would also create a risk that a simple change to taking 
//ABigInt as a const parameter could suddenly lead to Access Violations.
于 2013-09-15T09:30:18.343 回答