delphi - 我可以将缓冲区 + 大小转换为 TBytes 吗?
问题描述
给定一个缓冲区及其以字节为单位的大小,有没有办法将其转换为 TBytes 而无需复制它?
例子:
procedure HandleBuffer(_Buffer: PByte; _BufSize: integer);
var
Arr: TBytes;
i: Integer;
begin
// some clever code here to get contents of the buffer into the Array
for i := 0 to Length(Arr)-1 do begin
HandleByte(Arr[i]);
end;
end;
我当然可以复制数据:
procedure HandleBuffer(_Buffer: PByte; _BufSize: integer);
var
Arr: TBytes;
i: Integer;
begin
// this works but is very inefficient
SetLength(Arr, _BufSize);
Move(PByte(_Buffer)^, Arr[0], _BufSize);
//
for i := 0 to Length(Arr)-1 do begin
HandleByte(Arr[i]);
end;
end;
但是对于一个大缓冲区(大约 100 兆字节),这意味着我的内存需求增加了一倍,并且还花费了大量时间不必要地复制数据。
我知道我可以简单地使用 PByte 来处理缓冲区中的每个字节,我只对使用 TBytes 的解决方案感兴趣。
我认为这是不可能的,但我以前错了。
解决方案
不,这是不可能的(没有不合理的黑客攻击)。
问题是这TBytes = TArray<Byte> = array of Byte
是一个动态数组,非空动态数组的堆对象有一个包含数组引用计数和长度的标题。
一个接受TBytes
参数的函数,当给定一个指向字节数组的普通指针时,可能(正确地)尝试读取(不存在的)标题,然后你就遇到了严重的麻烦。
此外,动态数组是托管类型(如我提到的引用计数所示),因此您可能也会遇到问题。
但是,在您的特定示例代码中,您实际上根本没有使用数据的动态数组性质,因此您可以直接使用缓冲区:
procedure HandleBuffer(_Buffer: PByte; _BufSize: integer);
var
i: Integer;
begin
for i := 0 to _BufSize - 1 do
HandleByte(_Buffer[i]);
end;
推荐阅读
- c# - 有没有办法将 IAsyncEnumerable 转换为 Stream?
- javascript - 如何在另一个 div 中的标签输入上显示多选下拉值?
- python - 通过 google play api 发布新版本返回 500 内部错误
- c# - c# 转义 " 字符
- java - 无法查找主机 Cluster0-XXXXX.mongodb.net 的 TXT 记录
- javascript - HTML5 视频 - onclick 播放到全屏
- python - 如何根据python中其他列的计算将值附加到新列?
- algorithm - 递归调用实际上如何在合并排序中工作?
- python - 在 python 中,如何在 Windows 中获取当前用户 SID?
- flutter - 如何使用 Flutter 更改 TextFormField 中边框线的颜色