C++学习笔记:智能指针删除器

删除器

很多时候我们都用new来申请空间,用delete来释放。库中实现的各种智能指针,默认也都是用delete
来释放空间,但是若我们采用malloc申请的空间或是用fopen打开的文件,这时我们的智能指针就无法
来处理,因此我们需要为智能指针定制删除器,提供一个可以自由选择析构的接口,这样,我们的智能
指针就可以处理不同形式开辟的空间以及可以管理文件指针。
自定义智能指针的方式有两种,函数指针与仿函数(函数对象)。原理为利用RAII技术对类对象进行托管,在RAII类中定义指向类对象的指针以及删除器(函数指针,指向自定义的类对象或者函数入口地址均可),同时重载相关运算符。

#define  _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <iostream>
#include <functional>
using std::cout;
using std::endl;
using std::string;
using std::function;

//测试类
class Point
{
public:
	Point(int x, int y)
		: _ix(x), _iy(y)
	{
		cout << "Point(int,int)" << endl;
	}

	~Point() { cout << "~Point()" << endl; }

	void print() const {
		cout << "(" << _ix << "," << _iy << ")" << endl;
	}

private:
	int _ix, _iy;
};

class Foo
{
public:
	Foo() : _a(1) { cout << "Foo()" << endl; }
	~Foo() { cout << "~Foo()" << endl; }
private:
	int _a = 1;

};

//默认删除器,类形式
template <class T>
struct DefaultDeleter
{
	void operator()(T* p)
	{
		if (p)
		{
			delete p;
			p = nullptr;
			cout << "DefaultDeleter operator()(T* p)" << endl;
		}
	}
};

//对文件指针指定回收方式,函数形式
void MyfpCloser(FILE *fp)
{
	if (fp)
	{
		fclose(fp);
		cout << "void MyfpCloser(FILE *fp)" << endl;
	}
};

//智能指针删除器部分
template <class T>
class RAII
{
public:
	RAII(T* p, function<void(T*)> del = DefaultDeleter<T>()) : _p(p), _deleter(del) {}

	~RAII() {
		if (_p)
		{
			_deleter(_p);
			_p = nullptr;
		}
	}
	
	T* get() { return _p; }
	//重载相关运算符
	T* operator->() { return _p; }
	T& operator*() { return *_p; }
	//唯一托管
	RAII(const RAII&) = delete;
	RAII& operator=(const RAII&) = delete;

private:
	T* _p;
	//函数指针,可用实例化的对象或者函数入口地址赋值。
	function<void(T*)> _deleter;
};

void test()
{
	RAII<Point> raii(new Point(1, 1));
	raii->print();
	(*raii).print();
	/*
	结果
	Point(int,int)
	(1,1)
	(1,1)
	~Point()
	DefaultDeleter operator()(T* p)
	*/
}

void test1()
{
	//可通过模版的第二个参数,或者构造函数的第二个参数初始化删除器。
	//RAII<FILE,  MyfpCloser> fp(fopen("wuhan.txt", "a+"));
	RAII<FILE> fp(fopen("wuhan.txt", "a+"), MyfpCloser());
	string msg = "hello world\n";
	fwrite(msg.c_str(), 1, msg.size(), fp.get());
	//结果 void MyfpCloser(FILE *fp)
	
}

int main(void)
{
	//test();
	test1();
	return 0;
}

如果错误欢迎指正,后续更新智能指针误用

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值