C++ 中 ZeroMemory、memset

使用C/C++编程时,常使用ZeroMemory、memset或 “={0}”来对结构体对象进行初始化或清零。然而这三种方式都有各自的特点,使用时需谨慎,否则容易出现严重错误,本人今日解决一个导致宕机的bug,查了几小时,才发现是由同事乱用ZeroMemory所致。于是搜集资料,撰此文以共勉。

memset
void *memset(void *s,int ch,size_t n);是由C Run-time Library提供的提供的函数,作用是在一段内存块中填充某个给定的值,它是对较大的结构体或数组进行清零操作的一种最快方法。由于是语言层面提供,所以可跨平台使用。
示例:

char str[] = "almost every programmer should know memset!";
memset (str,'-',6);
puts (str);

输出:

------ every programmer should know memset!

ZeroMemory
ZeroMemory是美国微软公司的软件开发包SDK中的一个宏。其作用是用0来填充一块内存区域。定义式如下:

#define RtlZeroMemory(Destination,Length) memset((Destination),0,(Length))
#define ZeroMemory RtlZeroMemory

由此可见:

  1. ZeroMemory实际是用memset实现的。
  2. ZeroMemory只能用于windows平台。

注意:
ZeroMemory和memset且于清零时,会将结构中所有字节置0,如果结构体中有虚函数或结构体成员中有虚函数,则会将虚函数指针置0,如果后续程序调用虚函数,空指针很可能导致程序崩溃!

因此,有虚函数或成员中有虚函数的结构体初始化,一定要用构造函数来完成。另外,如果一个类的结构中包含STL模板(Vector、List、Map等等),那么使用ZeroMemory对这个类的对象中进行清零操作也会引起一系列的崩溃问题(指针指向内存错误、迭代器越界访问等)。所以,再次强烈建议:类(class)只使用构造函数进行初始化,不要调用ZeroMemory进行清零操作。
示例:

#include <iostream>
#include <Windows.h>

class Car
{
public:
	virtual void Run() {}
	void print() { puts("hello world"); }
private:
	int m_speed;
};
struct	SRace
{
	Car car;
	int dirver;
};

int main(void)
{
	char str[] = "almost every programmer should know memset!";
	memset(str, '-', 6);
	puts(str);

	SRace race;
	race.car.print();
	memset(&race, 0, sizeof(SRace));
	// ZeroMemory(&race, sizeof(race));

	race.car.print();
	// 没通过虚表指针调用,没事
	race.car.Run();

	Car* pCar = &race.car;

	// __vfptr = 0x00000000,崩溃
	pCar->Run();

	return 0;
}

类的多个对象共享类的函数,类中函数所占空间加载时将其放到了代码区,每个类的对象都会分配其非静态成员变量空间,我们可以根据对象中成员变量的不同取值确定对象的不同状态,sizeof(class_object)就是返回对象所占空间的大小。
细化一下:
代码区存储程序的代码
数据区存放全局变量和静态变量,与常量 其实可以细化为全局/静态数据区与常量数据区
栈中存放局部变量与传来的参数
堆是动态分配使用的。
类在内存里的数据连续存储,基本上就是存几个非静态成员变量与虚函数表指针(如果存在)

={0}
={0}操作是结构体和数组的一种初始化方式,它是将结构体中基本类型变量赋默认值,当结构体中有非基本类型(例如类对象)时,会编译错误,这也是一种保护。

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值