C++11/14之重载全局operator new和operator delete函数

重载全局operator new和operator delete函数,operator new[]和operator delete[]函数

在这里插入图片描述
在这里插入图片描述

  • void *operator new(size_t size)
    {
    return malloc(size);
    }
  • void *operator new[](size_t size) //数组版本
    {
    return malloc(size);
    }
  • void operator delete(void *phead)
    {
    free(phead);
    }
  • void operator delete[](void *phead)
    {
    free(phead);
    }
class A
	{
	public:
		int m_i;
		int m_j;
		A()
		{
			cout << "A::A()" << endl;
		}
		~A()
		{
			cout << "A::~A()" << endl;
		}


		void *operator new(size_t size)
		{
			A *ppoint = (A*)malloc(size);
			return ppoint;
		}
		void *operator new[](size_t size) //数组版本
		{
			A *ppoint = (A*)malloc(size);
			return ppoint;
		}
		void operator delete(void *phead)
		{
			free(phead);
		}
		void operator delete[](void *phead)
		{
			free(phead);
		}
	};

	void func()
	{
		int *pint = new int(12); //调用重载的operator new,给size传4字节
		delete pint;			//调用重载的operator delete

		char *parr = new char[10]; //调用重载的operator new[],给size传10字节
		delete[] parr;			//调用重载的operator delete[]


		A *p = new A();  //调用重载的operator new,给size传8字节
		delete p;        //调用重载的operator delete 

		A *pa = new A[3]();  /调用重载的operator new[],给size传28字节
		delete[] pa;		//调用重载的operator delete[]
		
		
	}

定位new(placement new)

有placement new,但是没有对应的placement delete
功能:在已经分配的原始内存中初始化一个对象;
a)已经分配,定位new并不分配内存,你需要提前将这个定位new要使用的内存分配出来
b)初始化一个对象(初始化一个对象的内存),我们就理解成调用这个对象的构造函数;

定位new就是能够在一个预先分配好的内存地址中构造一个对象

格式:new (地址) 类类型(参数)

  • b=void * mymemPoint = (void *)new char[sizeof(A)]; //内存必须事先分配出来

  • A *pmyAobj1 = new (mymemPoint) A(); //调用无参构造函数,这里并不会额外分配内存

  • void *mymemPoint2 = (void *)new char[sizeof(A)];

  • A *pmyAobj2 = new (mymemPoint2) A(12); //调用带一个参数的构造函数,这里并不会额外分配内存

class A
	{
	public:
		int m_a;
		A() :m_a(0)
		{
			int test;
			test = 1;
		}
		A(int tempvalue) :m_a(tempvalue)
		{
			int test;
			test = 1;
		}
		~A()
		{
			int abc;
			abc = 1;
		}
}


void *mymemPoint = (void *)new char[sizeof(A)]; //内存必须事先分配出来
A *pmyAobj1 = new (mymemPoint) A(); //调用无参构造函数,这里并不会额外分配内存
	
void *mymemPoint2 = (void *)new char[sizeof(A)];
A *pmyAobj2 = new (mymemPoint2) A(12); //调用带一个参数的构造函数,这里并不会额外分配内存

//传统方式释放可行
//delete pmyAobj1;
//delete pmyAobj2;
 
pmyAobj1->~A(); //手工调用析构函数是可以的,但手工调用构造函数一般不可以
pmyAobj2->~A();
delete[](void *)pmyAobj1;
delete[](void *)pmyAobj2;

定位new操作符的重载

void *operator new(size_t size,void *phead)
{
//干点别的事
return phead; //收到内存开始地址,只需要原路返回
}

多种版本的operator new重载

可以重载很多版本的operator new,只要每个版本参数不同就行,但是第一个参数是固定的,都是size_t,表示你要new对象的sizeof值

class A
{
public:
	void *operator new(size_t size, int tvp1, int tvp2)
	{
		return NULL;
	}
	A()
	{
		int test;
		test = 1;
	}
};
void func()
{
	A *pa =  new (1234, 56) A(); //自定义 new,不调用构造函数

}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++中,可以重载newdelete运算符以定制动态内存管理的行为。重载new运算符可以用于自定义内存分配的方式,而重载delete运算符可以用于自定义内存释放的方式。 重载new运算符的一种常见方式是定义一个全局new运算符函数,并使用该函数来执行内存分配。例如: ```cpp void* operator new(size_t size) { // 自定义内存分配逻辑 void* ptr = malloc(size); // 检查分配是否成功 if (ptr == nullptr) { throw std::bad_alloc(); } return ptr; } ``` 重载delete运算符的一种常见方式是定义一个全局delete运算符函数,并使用该函数来执行内存释放。例如: ```cpp void operator delete(void* ptr) noexcept { // 自定义内存释放逻辑 free(ptr); } ``` 需要注意的是,如果重载new运算符,通常也需要相应地重载delete运算符,以确保内存的正确释放。 可以根据需要重载其他版本的newdelete运算符,例如带有额外参数的newdelete运算符,数组形式的newdelete运算符等。重载这些运算符时需要遵循一定的规则和约定,确保正确性和可靠性。 值得注意的是,C++11引入了更加灵活和安全的内存管理方式,例如智能指针(如std::shared_ptr和std::unique_ptr)和RAII(资源获取即初始化)等,这些方式可以减少手动管理内存的复杂性和错误。因此,在使用newdelete运算符进行内存管理之前,建议先考虑这些更高级的内存管理工具。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值