我将代码从10.1移植到10.2,这给了我错误:
procedure TForm4.FormCreate(Sender: TObject);
const
CFourBytes: array[0..3] of Byte = (1, 2, 3, 4);
var
LStream: TMemoryStream;
LBuffer: array of Byte;
begin
SetLength(LBuffer, 4);
LStream := TMemoryStream.Create;
LStream.Write(@CFourBytes[0], 4); // E2036 Variable required
LStream.Position := 0;
LStream.ReadData(@LBuffer[0], 4);
end;
我必须将违规行更改为LStream.Write(CFourBytes[0], 4);
发生了什么变化?我一直都做错了吗?
答案 0 :(得分:3)
您问题中的代码确实在较旧版本中进行了编译,但不应这样做。在10.2中看到的行为是正确的。
在旧版本中发生的事情非常奇怪。编译器在TStream
中选择此重载:
function Write(const Buffer: TBytes; Count: Longint): Longint; overload;
这特别糟糕,因为传递给此方法的是静态数组CFourBytes
的地址。绝对不是TBytes
对象。
现在恰好发生了一个TBytes
变量是数组第一个元素的地址。 TBytes
的{{1}}替代中没有任何内容引用该伪造TMemoryStream.Write
对象的Length()
。因此,您的代码恰好按预期工作。很明显,这是一个已修复的编译器错误。
您的代码一直被破坏,直到现在,您还是幸运地摆脱了它。您应该修复您的代码。像这样:
TBytes
请注意,我使用的是LStream := TMemoryStream.Create;
try
LStream.WriteBuffer(CFourBytes, SizeOf(CFourBytes));
SetLength(LBuffer, LStream.Size);
LStream.Position := 0;
LStream.ReadBuffer(LBuffer[0], LStream.Size);
finally
LStream.Free;
end;
和WriteBuffer
而不是ReadBuffer
和Write
。这些是Read
首选的方法。原因是它们执行错误检查并在发生错误的情况下引发异常,这与TStream
和Write
不同。
答案 1 :(得分:1)
也许什么都没有改变。
TStream.Write/Read
方法始终使用未类型化的const/var
参数const Buffer
(help),并且使用变量地址是错误的(因为方法(准确地说,是编译器)找到了地址)变量本身)。
可能您不小心将这些方法与使用 typed 参数且其中overloaded versions之一获得Pointer
类型参数的Read / Write Data 方法混淆。
这里ReadData
实现取消引用此指针,并在内部使用Read
(Read
依次调用Move
,最后一个例程再次获取缓冲区的地址:))>