TBytes变量的正确使用模式是什么?根据我的理解,TBytes不是一个类,而是一个“动态字节数组”。我不确定在哪里释放内存,释放它时,哪个是从生产者传递给消费者的最佳方式。我希望我的生产者创建一个TBytes实例,然后将其传递给消费者。在这种情况发生之后,生产者想要重用其TBytes成员变量,知道消费者最终将内存返回给系统的内容。如果TBytes是一个对象,我不会有任何问题,但我不确定TBytes在这种情况下是如何工作的。
例如,在对象A中,我想将一些数据组装到TBytes数组中,该数组是对象A的成员。完成后,我想将TBytes数组传递给另一个对象B,然后该对象变为数据的所有者。同时,回到对象A,我想开始组装更多数据,重用TBytes成员变量。
type
TClassA = class
private
FData: TBytes;
public
procedure AssembleInput(p: Pointer; n: Cardinal);
end;
TClassB = class
public
procedure ProcessData(d: TBytes);
end;
var
a: TClassA;
b: TClassB;
procedure TClassA.AssembleInput(p: Pointer; n: Cardinal);
begin
SetLength(FData, n);
Move(p^, FData, n); // Is this correct?
...
b.ProcessData(FData);
...
// Would it be legal to reuse FData now? Perhaps by copying new (different)
// data into it?
end;
procedure TClassB.ProcessData(d: TBytes);
begin
// B used the TBytes here. How does it free them?
SetLength(d, 0); // Does this free any dynamic memory behind the scenes?
end;
提前致谢!
答案 0 :(得分:15)
Delphi动态数组是具有自动生命周期管理的托管类型。它们被引用计数,当引用计数变为0时,处理。你可以认为它们在字符串,接口和变体方面是等效的。
您可以通过以下三种方式之一显式释放对动态数组的引用:
a := nil;
Finalize(a);
SetLength(a, 0);
然而,只是不做任何事情并且在变量离开范围时释放引用是很常见的。
使用动态数组时要注意的一件事是,对同一动态数组有两个引用。在这种情况下,通过一个引用应用的更改在另一个引用中是可见的,因为只有一个对象。
SetLength(a, 1);
a[0] := 42;
b := a;
b[0] := 666;//now a[0]=666
你问这是否正确:
Move(p^, FData, n);
不,不是。您在此处所做的是将p
的内容复制到引用FData
上。如果您想使用Move
进行复制,那么您可以写下:
Move(p^, Pointer(FData)^, n);
或者如果你想更加冗长并避免演员阵容,你可以写:
if n>0 then
Move(p^, FData[0], n);
由于Move
完全没有类型安全,所以我个人对演员表现不太感兴趣。
现在重用FData是否合法?也许通过将新的(不同的)数据复制到其中?
如果没有更多背景,我觉得我无法回答这个问题。例如,我不知道为什么FData
是一个字段,因为它仅在本地用于该函数。作为局部变量会更有意义。据推测,它被宣布为一个领域是有原因的,但是不能从这段代码中轻易辨别出来。
您使用生产者/消费者模式。通常这样做是为了将生产与消费分离。但是,您的示例代码不会这样做,大概是因为解耦代码太复杂而不能包含在这里。
对于真正的生产者/消费者实施,您需要将数据的所有权从生产者转移到消费者。根据我们上面所描述的,一种非常简单有效的方法是使用引用计数。当数据传输给消费者时,生产者应该释放对它的引用。
答案 1 :(得分:4)
您的代码中存在一些误用。以下更正确:
type
TClassA = class
private
FData: TBytes;
public
procedure AssembleInput(p: Pointer; n: NativeUInt);
end;
TClassB = class
public
procedure ProcessData(var d: TBytes);
end;
var
a: TClassA;
b: TClassB;
procedure TClassA.AssembleInput(p: Pointer; n: NativeUInt);
begin
SetLength(FData, n);
if n <> 0 then Move(p^, FData[0], n);
...
b.ProcessData(FData);
// FData is ready for reuse here...
end;
procedure TClassB.ProcessData(var d: TBytes);
begin
...
SetLength(d, 0);
end;
答案 2 :(得分:-2)
移动(p ^,FData,n);这没关系
程序TClassB.ProcessData(d:TBytes); // d是FData的引用计数
开始
// d只保留FData,而refcount = 1则保持不变
//如果你在d前放置“var”关键字,FData将被释放
SetLength(d,0);
端;