FillMemory()、ZeroMemory(),fillchar()

 FillMemory、ZeroMemory 一目了然的两个函数, 但其实它们都是调用了 FillChar;

清空不过就是填充空字符(#0: 编号为 0 的字符), 说来说去是一回事.

为了下面的测试, 先写一个以十六进制方式查看内存的函数:
--------------------------------------------------------------------------------
 
function GetMemBytes(var X; size: Integer): string;
var
  pb: PByte;
  i: Integer;
begin
  pb := PByte(X);
  for i := 0 to size - 1 do
  begin
    Result := Result + IntToHex(pb^, 2) + #32;
    Inc(pb);
  end;
end; {GetMemBytes end}

//测试:
var
  p1: PAnsiChar;
  p2: PWideChar;
  s1: AnsiString;
  s2: UnicodeString;
begin
  p1 := 'ABCD';
  p2 := 'ABCD';
  s1 := 'ABCD';
  s2 := 'ABCD';

  ShowMessage(GetMemBytes(p1,4)); {41 42 43 44}
  ShowMessage(GetMemBytes(p2,8)); {41 00 42 00 43 00 44 00}
  ShowMessage(GetMemBytes(s1,4)); {41 42 43 44}
  ShowMessage(GetMemBytes(s2,8)); {41 00 42 00 43 00 44 00}
end;
--------------------------------------------------------------------------------

测试 FillMemory、ZeroMemory、FillChar 三个填充函数:
--------------------------------------------------------------------------------
 
const
  num = 10;
var
  p: PChar;
begin
  p := StrAlloc(num);

  ShowMessage(GetMemBytes(p, num)); {从结果看出 StrAlloc 没有初始化内存}

  FillMemory(p, num, Byte('A'));
  ShowMessage(GetMemBytes(p, num)); {41 41 41 41 41 41 41 41 41 41}

  ZeroMemory(p, num);
  ShowMessage(GetMemBytes(p, num)); {00 00 00 00 00 00 00 00 00 00}

  FillChar(p^, num, 'B');
  ShowMessage(GetMemBytes(p, num)); {42 42 42 42 42 42 42 42 42 42}

  StrDispose(p);
end;

此时, 我想到一个问题:
GetMem 和 GetMemory 没有初始化内存; AllocMem 会初始化内存为空, 那么
ReallocMem、ReallocMemory 会不会初始化内存?
测试一下(结果是没有初始化):
--------------------------------------------------------------------------------
 
{测试1}
var
  p: Pointer;
begin
  p := GetMemory(3);
  ShowMessage(GetMemBytes(p, 3));
  ReallocMem(p, 10);
  ShowMessage(GetMemBytes(p, 10)); {没有初始化}
  FreeMemory(p);
end;

{测试2}
var
  p: Pointer;
begin
  p := AllocMem(3);
  ShowMessage(GetMemBytes(p, 3));
  ReallocMem(p, 10);
  ShowMessage(GetMemBytes(p, 10)); {没有初始化}
  FreeMemory(p);
end;

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
在 C# 中,可以使用 Marshal 类的静态方法 ZeroMemory() 或 Copy() 方法来实现 FillMemory() 的功能。 ZeroMemory() 方法可以将指定地址的内存空间清零,其函数签名如下: ```csharp public static void ZeroMemory(IntPtr destination, int length); ``` 其中,第一个参数是指向要清零的内存空间的指针,第二个参数是要清零的字节数。例如,下面的代码片段可以将一个长度为 100 的字节数组 a 的前 10 个字节清零: ```csharp byte[] a = new byte[100]; GCHandle handle = GCHandle.Alloc(a, GCHandleType.Pinned); IntPtr ptr = handle.AddrOfPinnedObject(); Marshal.ZeroMemory(ptr, 10); handle.Free(); ``` 上面的代码中,首先使用 GCHandle.Alloc() 方法将字节数组 a 固定在内存中,然后使用 AddrOfPinnedObject() 方法获取其地址,最后调用 ZeroMemory() 方法将前 10 个字节清零。需要注意的是,使用 GCHandle 固定内存后,应该在使用完后调用 GCHandle.Free() 方法将其释放。 另外,Copy() 方法也可以实现 FillMemory() 的功能,其函数签名如下: ```csharp public static void Copy(byte[] source, int sourceIndex, IntPtr destination, int destinationIndex, int length); ``` 其中,第一个参数是源字节数组,第二个参数是源字节数组的起始索引,第三个参数是目标内存地址,第四个参数是目标内存地址的起始索引,第五个参数是要复制的字节数。例如,下面的代码片段也可以将一个长度为 100 的字节数组 a 的前 10 个字节清零: ```csharp byte[] a = new byte[100]; GCHandle handle = GCHandle.Alloc(a, GCHandleType.Pinned); IntPtr ptr = handle.AddrOfPinnedObject(); byte[] zeros = new byte[10]; Marshal.Copy(zeros, 0, ptr, 10); handle.Free(); ``` 上面的代码中,首先使用 GCHandle.Alloc() 方法将字节数组 a 固定在内存中,然后使用 AddrOfPinnedObject() 方法获取其地址,接着创建一个长度为 10 的全零字节数组 zeros,最后调用 Copy() 方法将其复制到 a 的前 10 个字节中。同样需要注意的是,使用 GCHandle 固定内存后,应该在使用完后调用 GCHandle.Free() 方法将其释放。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值