答案:memset按字节初始化内存,仅适用于POD类型且仅对0值安全,非0值或复杂对象需用fill或构造函数。
在C++中,memset 是一个来自 C 标准库的函数,常用于内存块的初始化或清零操作。虽然使用简单,但在实际开发中容易因误解其行为而导致严重问题。下面介绍 memset 的常见用法及使用时的关键注意事项。
void* memset(void* ptr, int value, size_t num);
作用是将从 ptr 指向的内存地址开始的 num 个字节,设置为 value(按字节赋值)。注意:value 虽然是 int 类型,但只会取低8位(即0~255)作为填充字节。常见用途如清零原始数组:
int arr[100]; memset(arr, 0, sizeof(arr)); // 正确:将所有字节设为0
例如:
int arr[10]; memset(arr, 1, sizeof(arr)); // 错误!不是每个元素变为1这会将每个字节设为 1,对
于 int 类型(通常4字节),每个 int 的二进制变成 0x01010101,即十进制 16843009,而不是1。只有当目标值为0时,这种操作才是安全且有效的。
例如:
std::string str = "hello"; memset(&str, 0, sizeof(str)); // 危险!破坏了 string 内部结构 // 后续调用 str.length() 可能崩溃这会清空对象的所有字节,包括 vtable 指针、内部缓冲区指针等,导致未定义行为。
例如:
int flags[10]; memset(flags, -1, sizeof(flags)); // 可能有效,但不推荐它确实会让每个字节变为 0xFF,从而使每个 int 成为 -1(补码下),但该行为不可移植,且可读性差。应改用显式赋值:
std::fill(flags, flags + 10, -1);
基本上就这些。memset 不复杂但容易忽略细节,尤其在混合C/C++代码中更需谨慎对待。