iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++入门之内存处理详解
  • 386
分享到

C++入门之内存处理详解

2024-04-02 19:04:59 386人浏览 八月长安
摘要

目录前言C/C++内存分布C语言中动态内存管理方式c++内存管理方式new和delete操作基础类型new和delete操作自定义类型基于malloc开辟并初始化的自定义类

前言

兜兜转转,我们终于结束了C++中非常重要的一环**(类和对象),现在来到了C++中的内存管理章节.在此篇文章中,博主将会介绍内存的分布,不同于c的新型申请堆区空间方法,new,delete和C中的malloc等有什么不同.**

C/C++内存分布

在c和c++中,内存区大概分为这几个板块:栈区,内存映射段,堆区,数据段和代码段.

  • 栈区: 存放非静态局部变量,函数参数,函数返回值等,其优先使用高地址,并逐渐往下.
  • 内存映射段:高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口创建共享内存,做进程间通信.由于博主还未更新到操作系统,这里不做过多介绍.
  • 堆区: 用于程序运行时进行动态内存分配(一般使用malloc),其优先使用低地址,逐渐往上.
  • 数据段:存储全局数据和静态变量.
  • 代码段:可执行的代码/只读常量.

理论千遍,不如用例子一现,大家往下看:

image-20211024195521282

在上图中,大家可以清晰的看到各种类型数据的存储区域,一目了然.

c语言中动态内存管理方式

我们在学习c语言时候,想要向堆区申请一块空间,只能通过malloc()函数,而且操作比较麻烦,需要计算申请空间的大小并且进行强制转换.


int* p1 = (int*) malloc(sizeof(int));
free(p1);
int* p2 = (int*)calloc(4, sizeof (int));
int* p3 = (int*)realloc(p2, sizeof(int)*10);
free(p3 );

我们可以清晰的看到,在c语言中申请一块堆区内存空间,操作有点繁琐,那么在C++语言中,是怎么申请一块堆区内存呢?

C++内存管理方式

在c++中,c语言的内存管理方式依然可以使用,但是有些地方仍然使用c中的方法进行申请就比较麻烦,因此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理.

那么什么时候使用c内存管理方式就会比较麻烦呢?比如下面:


class Stack
{
public:
	Stack(int* p,int n)
    {
        val = p;
        top = capacity = n;
    }
private:
    int* val;
    int top;
    int capacity;
};
int main()
{
    Stack* stack = (Stack*)malloc(sizeof(Stack));
    //然后现在我们想要进行初始化,发现好像不能操作了(因为私有成员外部无法访问),这就会比较麻烦,因此提出了new和delete的操作
    return 0;
}

new和delete操作基础类型

①开辟单个元素基本语法: type* name = new type(content);

①释放空间语法: delete name;

其中type是开辟的元素类型,name是变量名,content是想要赋的值,例如:


int* a = new int(10);         //开辟一个整型空间,并且初始化为10;
char* s = new char('s');      //开辟一个字符空间,并且初始化为's';
delete a;
delete s;                     //释放a和s

②开辟数组基本语法: type* name = new type[n]

②删除数组基本语法:delete[] name;

其中type是开辟的元素类型,name是数组名,n是空间数量,例如:


int* num = new int[10];         //开辟10个int类型的空间
char* str = new char[10];        //开辟10个char类型的空间
delete[] num;
delete[] str;    //释放num和str;

注意点:在c语言中malloc,realloc,calloc等是函数,在c++中,new和delete是操作符.

new和delete操作自定义类型

使用语法和上面的自定义类型几乎一样,只是初始化位置有点差别,语法:type* name = new type(s1,s2,s3...);

其中type是自定义的类型,s1,s2,s3等是自定义类型中构造函数的对应参数.例子:


class Test
{
public:
    Test(int a,int b,int c)
    {
        _a = a;
        _b = b;
        _c = c;
    }
private:
    int _a;
    int _b;
    int _c;
};
int main()
{
    Test* t = new Test(1,2,3);  //根据构造函数参数列表写对应参数.
    delete t;
}

基于malloc开辟并初始化的自定义类型

有人可能会有点好奇,说,我就是要用malloc开辟自定义类型,但是我还想初始化,这么进行操作?答案是,可以的,需要搭配new

语法 new(type*) type(s1,s2,s3...)

什么意思呢?我们仍然按照上面的Test类举例:


Test* t = (Test*) malloc(sizeof(Test));
new(t)Test(1,2,3);    //这样就可以初始化了.

new和delete底层实现原理

在讲解他们的底层实现原理之前我们先介绍一下两个全局函数,分别是operator newoperator delete.

operator new和operator delete

大家看到上面的形式可能会误认为是对new和delete进行了重载,实际并不是,只是这两个函数就叫做这名字而已.

我们在学习C语言时候,还记得是当开辟空间时候需要进行检查是否成功吗?而operator new其实和malloc一模一样,只是它对空间开辟失败后做出的反应是抛出异常,而c语言中,我们是手动判断,然后停止.下面是operator new的代码:


void* __CRTDECL operator new(size_t size) _THROW1(_STD bad_alloc)
{
	// try to allocate size bytes
	void* p;
	while ((p = malloc(size)) == 0)              //如果开辟成功就不会进入循环,并且可以清晰
		if (_callnewh(size) == 0)
		{
			// report no memory
			// 如果申请内存失败了,这里会抛出bad_alloc 类型异常
			static const std::bad_alloc nomem;
			_RAISE(nomem);
		}
	return (p);
}

同理,operator delete不过也就是free,下面是它的代码:


void operator delete(void* pUserData)
{
	_CrtMemBlockHeader* pHead;
	RTCCALLBACK(_RTC_Free_hook, (pUserData, 0));
	if (pUserData == NULL)
		return;
	_mlock(_HEAP_LOCK); 
	__TRY
		
		pHead = pHdr(pUserData);
	
	_ASSERTE(_BLOCK_TYPE_IS_VALID(pHead->nBlockUse));
	_free_dbg(pUserData, pHead->nBlockUse);
	__FINALLY
		_munlock(_HEAP_LOCK); 
	__END_TRY_FINALLY
		return;
}

总结:operator new其实就是对malloc的封装,operator delete 就是堆free的封装.

new的底层实现

new的实现其实是进一步对malloc的封装,因为new的操作可以分为下面两件事:

  • 调用operator new进行开辟空间.
  • 如果开辟的空间是自定义类型,new再调用其构造函数.

所以说,new的内部其实就是有构造函数和operator new封装而成.

delete的底层实现

同样的道理,delete不过就是对free的进一步封装,因为delete的操作可以分为下面两件事:

  • 如果new开辟的空间是自定义类型,则首先调用其析构函数释放其内部资源.
  • 然后再调用operator delete,进行释放new所开辟出来的空间.

注意了,这里可能有人不明白自定义类型先释放内部,然后销毁外部空间啥意思,博主这里画图介绍,不过大家先看一下下面的一个类:


class Stack
{
public:
	Stack()
		:num(new int[10]),
		top(0),capacity(10)
	{}
	~Stack()
	{
		delete[] num;
		top = capacity = 0;
	}
private:
	int* num;
	int top;
	int capacity;
};
int main()
{
    Stack stack;
    delete stack;
    return 0;
}

如果我们定义一个该对象,那么其空间分布如下:

image-20211025172631194

如果delete不先调用析构函数,那么释放了stack对象后,在堆区里面的num将会继续存在,导致内存泄露.

new[]的底层实现

1.调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对象空间的申请

2.在申请的空间上执行N次构造函数

delete[]的原理

1.在释放的对象空间上执行N次析构函数,完成N个对象中资源的清理

2.调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释放空间

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注编程网的更多内容!

--结束END--

本文标题: C++入门之内存处理详解

本文链接: https://www.lsjlt.com/news/157722.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • C++入门之内存处理详解
    目录前言C/C++内存分布c语言中动态内存管理方式C++内存管理方式new和delete操作基础类型new和delete操作自定义类型基于malloc开辟并初始化的自定义类...
    99+
    2024-04-02
  • JVM入门之JVM内存结构内容详解
    一、java代码编译执行过程 源码编译:通过Java源码编译器将Java代码编译成JVM字节码(.class文件) 类加载:通过ClassLoader及其子类来完成...
    99+
    2024-04-02
  • C语言入门篇--理解地址及内存
    1.内存 内存是电脑中一个重要的存储器,计算机中所有的程序都在内存中运行的,内存的性能对计算机的影响非常大。 内存是计算机与CPU进行沟通的桥梁,计算机会把程序由硬...
    99+
    2024-04-02
  • C++入门之vector使用详解
    目录前言创建对象迭代器数据插入数据删除容量操作总结前言 兜兜转转,我们来到了C++的vector章节,今天就讲讲怎么使用vector吧. vector的本质就是一个线性的顺序表,只不...
    99+
    2024-04-02
  • 详解C/C++内存管理
    目录C/C++内存分布C语言中动态内存管理方式C++中动态内存管理方式new和delete操作内置类型new和delete操作自定义类型operator new和operator d...
    99+
    2024-04-02
  • C++入门教程之内联函数与extern "C"详解
    目录一.   内联函数1.概念及分析2.特性3.宏二.   extern “C”1.C++程序2.C程序 总...
    99+
    2023-01-10
    c++内联函数是什么 c++  extern c c++内联函数使用
  • C++入门之list的使用详解
    目录前言构造的使用1 构造空list2 构造含n个值为val的元素3 拷贝构造4 用迭代区间迭代器接口1 正常迭代接口2 逆向迭代接口容量接口元素访问数据修改头插头删尾插尾删pos位...
    99+
    2024-04-02
  • 详解C语言之动态内存管理
    目录开辟动态内存的函数释放开辟的动态内存空间的函数错误信息函数具体使用例: 常见的动态内存错误总结先来了解一下动态管理内存所需用到的函数 开辟动态内存的函数 1.mallo...
    99+
    2024-04-02
  • Spring Boot学习入门之AOP处理请求详解
    前言面向切面(AOP)Aspect Oriented Programming是一种编程范式,与语言无关,是一种程序设计思想,它也是spring的两大核心之一。在spring Boot中,如何用AOP实现拦截器呢?首先加入依赖关系:<d...
    99+
    2023-05-31
    springboot aop 处理请求
  • C/C++深入讲解内存管理
    目录C/C++内存分布C语言中的动态内存管理C++的内存管理operator new与operator delete函数operator new与operator dele...
    99+
    2024-04-02
  • C++嵌入式内存管理详情
    目录一、Linux内核系统结构二、查看Linux内存1.cache2.buffer三、内存补齐前言: 上一篇介绍了软件层面上的内存,并没有涉及很多底层的原理;但在实际工程中,部署一个...
    99+
    2024-04-02
  • C++之内存泄漏排查详解
    目录一 、经验排查二 、使用Visual Leak Detector for Visual C++2.1 Visual Leak Detector for Visual C++简介2...
    99+
    2024-04-02
  • 怎么理解C++内存处理
    这篇文章主要介绍“怎么理解C++内存处理”,在日常操作中,相信很多人在怎么理解C++内存处理问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么理解C++内存处理”的疑惑有所帮助!接下来,请跟着小编一起来学习吧...
    99+
    2023-06-25
  • Spring Boot学习入门之统一异常处理详解
    前言关于之前的一篇所讲到的表单验证中提到,如果产生错误,可以得到错误的信息,但是返回值的问题却没有考虑。其中所提到的Controller:@RequestMapping(value = "/doRegister", method = Req...
    99+
    2023-05-31
    spring boot 统一异常处理
  • C++入门之vector的底层实现详解
    目录前言定义初始结构声明构造函数容量有关操作获取有效数据大小size()获取数据容量capacity()增加容量reserve()重置大小resize()迭代器数据操作尾插push_...
    99+
    2024-04-02
  • C语言入门之基础知识详解
    目录一、思维导图二、环境搭建三、编码规范四、基本数据类型五、分支循环5.1 分支5.2 循环六、break与continue6.1 无限循环一、思维导图 内容不限于此思维导图 二、...
    99+
    2024-04-02
  • C++入门语法之函数重载详解
    目录写在前面1 函数重载的概念2 函数重载原理总结写在前面 关于C语言的编译与链接不懂的可以看一下下面的文章,先回顾一下以前的知识。 详解C语言的编译与链接 1 函数重载的概念 函数...
    99+
    2024-04-02
  • 详解Unity入门之GameObject
    目录Input和InputManagerMonoBehaviour总结GameObject和Component GameObject是游戏场景中真实存在的,而且有位置的一个物件 Co...
    99+
    2024-04-02
  • Java之MyBatis入门详解
    目录一、三层架构Java中的三层架构指(1)三层对应的包(2)三层中类的交互(3)三层对应的处理框架(4)使用三层架构有点(5)缺点二、MyBatis1、官方简介2、百度基本信息&n...
    99+
    2022-12-21
    Java MyBatis入门 Java MyBatis MyBatis入门
  • C++入门笔记之std::vector容器详解
    目录前言1. vector的构造函数原型: 2. vector的赋值函数原型:3. vector的容量和大小函数原型:4. vector的插入和删除函数原型:5. vector的存取...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作