您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > C/C++/C#

C++|封装裸指针为像指针的类(pointer-like class)

时间:2019-12-12 12:02:07  来源:  作者:

指向堆内存的裸指针就像一把没有刀鞘的利刃,功能强大但又有安全隐患。裸指针封装成类,就如同设计一个刀鞘。封装类可以让其有更多的自定义行为,处于你设想的可控范围内。这种封装的类就称为像指针的类(pointer-like class),一般都包含有一个裸指针,和一系列的成员函数(包括运算符重载)。根据成员函数的不同,这些pointer-like class可以是智能指针,也可以是容器的迭代器。

new与delete需要成对使用,以避免内存泄漏,但当有异常出现时呢?当在函数中返回堆内存指针,需要由调用者释放时,很容易忘记delete。

考虑一个动态分配值的函数:

void someFunction(){ Resource *ptr = new Resource; // Resource is a struct or class // do stuff with ptr here delete ptr;}

尽管上面的代码看起来相当简单,但是很容易忘记释放ptr。即使您确实记得在函数的末尾删除PTR,但是如果函数早退出,就有无数种方法可以不删除PTR。这可以通过提前返回来实现或者通过抛出异常:

#include <IOStream>
 
void someFunction()
{
 Resource *ptr = new Resource;
 
 int x;
 std::cout << "Enter an integer: ";
 std::cin >> x;
 
 if (x == 0)
 throw 0; // the function returns early, and ptr won’t be deleted!
 
 // do stuff with ptr here
 
 delete ptr;
}

在上述程序中,执行早期的return或throw语句,导致函数终止而不删除变量ptr。因此,分配给变量ptr的内存现在泄漏了(每次调用此函数并提前返回时,都会再次泄漏)。

从本质上讲,这些问题的发生是因为指针变量本身没有内在的机制来清理它们。

当函数返回指向堆内存的指针给调用者时:

Gadget* f(int n)
{
 // ... 
 auto p = new Gadget{n};
 return p;
}
void foo(n)
{
 auto p = f(n);
 // ……
 delete p; // easily forgot
}

上面的程序如果调用函数没有忘记,可以完成堆内存释放,但违背了“谁申请,谁释放”的原则,不是最佳的模块设计。

我们知道,局部变量(存储在栈上)在超出作用域时会自动释放。而类对象的机制增加了一个自动析构的析构函数,即局部对象(存储在栈上)不但自动释放内存,而且还自动执行一个析构函数,通常,我们会在析构函数中定义释放堆内存的行为。如果一个裸指针指向堆内存,就存在一个需要释放堆内存的行为,如果将此裸指针封装,释放的动作放在析构函数内,那此释放的动作就会自动化了,这就是裸指针封装的原因:

template<typename T>
class SP
{
public:
	SP():p(NULL){}
	SP(int n){p = new char[n];}
	~SP(){ delete[] p;}
	T& operator*() const
	{ return *px; }
	T* operator->() const
	{ return px; }
	SP(T* p):px(p){}
	// 如果用做迭代器要重载++、--、+n、-n等运算符
private:
	T* px;
	long* pn;
};

另外,当多个指针指向同一块堆内存时,怎样delete?如何避免出错?如指针的复制、移动行为,指针形参与指针实参结合时:

void passByValue(Auto_ptr1<Resource> res)
{
}
 
int main()
{
	Auto_ptr1<Resource> res1(new Resource);
	passByValue(res1)
 
	return 0;
}

在这个程序中,res1将被value复制到passByValue的参数res中,从而导致资源指针的复制。多个指针指向同一块堆内存,一个指针的delete行为势必影响到另一个指针。

很明显这不好。我们如何解决这个问题?

移动语义或unique_ptr或shared_ptr。

对于文件句柄,同样的,封装的文件流对象具有更好的安全性:

 void f(const string& name)
{
 FILE* f = fopen(name, "r"); // open the file
 vector<char> buf(1024);
 auto _ = finally([f] { fclose(f); }); // remember to close the file
 // ...
}

以上实例,buf的内存分配可能失败,然后文件句柄产生泄露。

以下实例使用则用封装的文件流对象,则可最大限度避免:

 
于void f(const string& name)
{
 ifstream f{name}; // open the file
 vector<char> buf(1024);
 // ...
}

同样的,也可以将指向堆内存的裸指针封装为一个迭代器,让其能够方便、安全地操作容器元素:

要自定义一个迭代器,就要重载一些基本操作符:*(解引用)、++(自增)、==(等于)、!=(不等于)、=(赋值)等。

#include <iostream>
using namespace std;
template <typename T>
class linkedList
{
private:
	struct node
	{
		int data;
		node * next;
	};
	typedef node* NODEPTR;
	
public:
	NODEPTR head;
	linkedList():head(NULL){}
	linkedList& push_front(int e)
	{
		NODEPTR p = new node;
		p->data = e;
		p->next = NULL;
		if(head!=NULL)
			p->next = head;
		head = p;
		return *this;
	}
	typedef NODEPTR _range;
	class iterator
	{
	private:
		_range p;
	public:
		iterator():p(NULL){}
		iterator(const linkedList& ll):p(ll.head){}
		iterator & operator+(const iterator& itr)
		{
			p=itr.p;
			return *this;
		}
		iterator& operator++()
		{
			p = p->next;
			return *this;
		}
		bool operator != (const iterator& itr)
		{
			return p!= itr.p;
		}
		T& operator *()
		{
			return p->data;
		}
	};
	iterator begin()
	{
		return iterator(*this);
	}
	iterator end()
	{
		return iterator();
	}
};
int main()
{ 
	linkedList<int> list;
	for(int i=0;i<=5;i++)
		list.push_front(i);
	
	for(linkedList<int>::iterator itr = list.begin();
			itr!=list.end();++itr)
		cout<<*itr<<' ';
	cin.get();
 return 0;
}
// 5 4 3 2 1 0

 



Tags:C++   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
一、编程语言1.根据熟悉的语言,谈谈两种语言的区别?主要浅谈下C/C++和PHP语言的区别:1)PHP弱类型语言,一种脚本语言,对数据的类型不要求过多,较多的应用于Web应用开发,现在好多互...【详细内容】
2021-12-15  Tags: C++  点击:(17)  评论:(0)  加入收藏
函数调用约定(Calling Convention),是一个重要的基础概念,用来规定调用者和被调用者是如何传递参数的,既调用者如何将参数按照什么样的规范传递给被调用者。在参数传递中,有两个很...【详细内容】
2021-11-30  Tags: C++  点击:(19)  评论:(0)  加入收藏
一、为什么需要使用内存池在C/C++中我们通常使用malloc,free或new,delete来动态分配内存。一方面,因为这些函数涉及到了系统调用,所以频繁的调用必然会导致程序性能的损耗;另一...【详细内容】
2021-11-17  Tags: C++  点击:(38)  评论:(0)  加入收藏
C++编程中,你是否有为 我到底该写个struct还是class 而苦恼过?如果你到现在还不知道该如何选择,那么请求继续阅读,下文或许能给你些建议。问题的产生C++语言继承了 C语言的 stru...【详细内容】
2021-10-18  Tags: C++  点击:(63)  评论:(0)  加入收藏
C++在C的面向过程概念的基础上提供了面向对象和模板(泛型编程)的语法功能。下面以一个简单实例(动态数组的简单封装,包括下标的值可以是任意正数值,并提供边界检查)来说明C++是如...【详细内容】
2021-10-18  Tags: C++  点击:(50)  评论:(0)  加入收藏
0 前言Hello,大家好,欢迎来到『自由技艺』的 C++ 系列专题。代码重用,尽可能避免冗余代码是程序员的一项必备技能,今天就来给大家介绍其中一种:函数装饰器。在设计模式中,与它对应...【详细内容】
2021-09-28  Tags: C++  点击:(75)  评论:(0)  加入收藏
今天我们就来聊一聊 C++ 中的异常机制吧。在学校期间,我们很少会用得上异常机制。然而,工作之后,很多时候却不得不引入异常机制。因为一般情况下,使用函数的返回值来确定函数的...【详细内容】
2021-09-26  Tags: C++  点击:(182)  评论:(0)  加入收藏
一、内存泄漏(memory leak)内存泄漏(memory leak)是指由于疏忽或错误造成了程序未能释放掉不再使用的内存的情况。内存泄漏并非指内存在物理上的消失,而是应用程序分配某段内存...【详细内容】
2021-09-03  Tags: C++  点击:(105)  评论:(0)  加入收藏
stack容器#include <iostream>using namespace std;#include <stack>//容器头文件void test(){stack<int>p;p.push(100);p.push(1000);p.push(100);while(!p.empty()){cout<...【详细内容】
2021-08-17  Tags: C++  点击:(81)  评论:(0)  加入收藏
stl 常用遍历算法(for_each transform)示例代码(结论在结尾!!!!)#include<iostream>using namespace std;#include"vector"#include"map"#include"string"#include"list"#in...【详细内容】
2021-08-13  Tags: C++  点击:(89)  评论:(0)  加入收藏
▌简易百科推荐
一、简介很多时候我们都需要用到一些验证的方法,有时候需要用正则表达式校验数据时,往往需要到网上找很久,结果找到的还不是很符合自己想要的。所以我把自己整理的校验帮助类分...【详细内容】
2021-12-27  中年农码工    Tags:C#   点击:(2)  评论:(0)  加入收藏
引言在学习C语言或者其他编程语言的时候,我们编写的一个程序代码,基本都是在屏幕上打印出 hello world ,开始步入编程世(深)界(坑)的。C 语言版本的 hello world 代码:#include <std...【详细内容】
2021-12-21  一起学嵌入式    Tags:C 语言   点击:(11)  评论:(0)  加入收藏
读取SQLite数据库,就是读取一个路径\\192.168.100.**\position\db.sqlite下的文件<startup useLegacyV2RuntimeActivationPolicy="true"> <supportedRuntime version="v4.0"/...【详细内容】
2021-12-16  今朝我的奋斗    Tags:c#   点击:(21)  评论:(0)  加入收藏
什么是shellshell是c语言编写的程序,它在用户和操作系统之间架起了一座桥梁,用户可以通过这个桥梁访问操作系统内核服务。 它既是一种命令语言,同时也是一种程序设计语言,你可以...【详细内容】
2021-12-16  梦回故里归来    Tags:shell脚本   点击:(18)  评论:(0)  加入收藏
一、编程语言1.根据熟悉的语言,谈谈两种语言的区别?主要浅谈下C/C++和PHP语言的区别:1)PHP弱类型语言,一种脚本语言,对数据的类型不要求过多,较多的应用于Web应用开发,现在好多互...【详细内容】
2021-12-15  linux上的码农    Tags:c/c++   点击:(17)  评论:(0)  加入收藏
1.字符串数组+初始化char s1[]="array"; //字符数组char s2[6]="array"; //数组长度=字符串长度+1,因为字符串末尾会自动添&lsquo;\0&lsquo;printf("%s,%c\n",s1,s2[2]);...【详细内容】
2021-12-08  灯-灯灯    Tags:C语言   点击:(47)  评论:(0)  加入收藏
函数调用约定(Calling Convention),是一个重要的基础概念,用来规定调用者和被调用者是如何传递参数的,既调用者如何将参数按照什么样的规范传递给被调用者。在参数传递中,有两个很...【详细内容】
2021-11-30  小智雅汇    Tags:函数   点击:(19)  评论:(0)  加入收藏
一、问题提出问题:把m个苹果放入n个盘子中,允许有的盘子为空,共有多少种方法?注:5,1,1和1 5 1属同一种方法m,n均小于10二、算法分析设f(m,n) 为m个苹果,n个盘子的放法数目,则先对...【详细内容】
2021-11-17  C语言编程    Tags:C语言   点击:(49)  评论:(0)  加入收藏
一、为什么需要使用内存池在C/C++中我们通常使用malloc,free或new,delete来动态分配内存。一方面,因为这些函数涉及到了系统调用,所以频繁的调用必然会导致程序性能的损耗;另一...【详细内容】
2021-11-17  深度Linux    Tags:C++   点击:(38)  评论:(0)  加入收藏
OpenCV(Open Source Computer Vision Library)是一个(开源免费)发行的跨平台计算机视觉库,可以运行在Linux、Windows、Android、ios等操作系统上,它轻量级而且高效---由一系列...【详细内容】
2021-11-11  zls315    Tags:C#   点击:(50)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条