当前位置: 首页 > wzjs >正文

宜昌网站建设网盘搜索引擎

宜昌网站建设,网盘搜索引擎,济南可信网站,百度舆情系统文章目录1、内存分布2、动态内存管理2.1 C中的动态内存管理2.2 C中的内存管理2.3 new和delete操作自定义类型2.4 new和delete的底层实现2.6 new和delete的实现原理2.7 定位new表达式的使用3、初识模板3.1 函数模板3.2 类模板1、内存分布 注意的是: 栈区是向低地址(…

文章目录

    • 1、内存分布
    • 2、动态内存管理
      • 2.1 C中的动态内存管理
      • 2.2 C++中的内存管理
      • 2.3 new和delete操作自定义类型
      • 2.4 new和delete的底层实现
      • 2.6 new和delete的实现原理
      • 2.7 定位new表达式的使用
    • 3、初识模板
      • 3.1 函数模板
      • 3.2 类模板

1、内存分布

在这里插入图片描述

注意的是:

  1. 栈区是向低地址(向下增长)开辟,堆区是向高地址(向上赠长)开辟。
  2. 未初始化的或初始化为0的变量(静态和全局的)存放在BSS段,未初始化的局部变量存放在栈区。
  3. 初始化的变量(静态和全局的)存放在数据段(也称为静态区)。
  4. 代码段(也称为常量区)存放程序编译后的汇编代码以及常量。
  5. 堆区只能动态开辟,栈区既可以静态开辟也可以动态开辟(_alloc函数)。

2、动态内存管理

2.1 C中的动态内存管理

在之前C语言中,我们一般用malloc函数进行动态内存开辟。
一般都有的步骤是 调用函数动态开辟、判断返回值、释放空间。

int* a = (int*)malloc(sizeof(int));
if(a == NULL)
{perror("malloc fail");exit(-1);
}
free(a);

2.2 C++中的内存管理

C中的内存管理在C++中也能继续使用。

C++规定通过一个new操作符来动态开辟空间,通过delete操作符来释放空间。

//动态申请一块int大小的空间
int* p1 = new int;//动态申请一块int大小的空间并初始化
int* p2 = new int(10);//动态申请多块int大小的空间并初始化
int* p3 = new int[10]{0};//释放空间
delete p1;
delete p2;//对多块空间释放,需要加[],[]内可以不写具体数字
delete[] p3;

注意new和delete操作符匹配,new[]和delete[] 操作符匹配。


那C++的动态开辟和C语言的动态开辟有什么区别呢?
下面我们继续探讨

2.3 new和delete操作自定义类型

先弄一个类A

#include <iostream>
using std::cout;
using std::endl;class A
{
public:A(int a = 1):_a(a){cout << "A(int a = 1)" << endl;}~A(){cout << "~A()" << endl;}
private:int _a;
};int main()
{A* a1 = (A*)malloc(sizeof(A));free(a1);cout << "-----" << endl;A* p1 = new A(1);delete p1;return 0;
}

在这里插入图片描述
首先可以明显的看到,new/delete和malloc/free的一个很大区别就是,对于类类型new/delete会调用构造函数和析构函数。
实际上:

  1. 对于内置类型,new/delete和malloc/free几乎一样。
  2. 对于自定义类型,new开了空间后还会调用构造函数,delete调用析构函数后会释放空间。

2.4 new和delete的底层实现

new的底层实现
下面是官方实现的operator new函数,从中大概来看,其实new的底层就是malloc,只是多个当申请失败会抛异常(这里先知道会抛异常,不用知道是什么)。

这里值得注意的是,operator new只是一个库里面实现的全局函数,并不是new的操作符重载,因为参数里面没有自定义类型。(这是一个对新手的误区,以为是操作符重载)

void *__CRTDECL operator new(size_t size) _THROW1(_STD bad_alloc)
{
void *p;
while ((p = malloc(size)) == 0) //malloc返回null就进入循环if (_callnewh(size) == 0){// 如果申请内存失败了,这里会抛出bad_alloc 类型异常static const std::bad_alloc nomem;_RAISE(nomem);}
return (p);
}

从底层代码我们可以了解new开辟失败返回异常,我们再来看看new开辟失败p1还会像malloc一样返回null指针吗。

#include<iostream>
using namespace std;
void Test1()
{while (1){//循环每次开辟空间 看看当开辟失败的时候p1返回是否为空char* p1 = new char[1024*1024*1024];if (p1 == nullptr){perror("malloc fail");exit(-1);}}
}int main()
{try{Test1();}catch (exception& e){cout << e.what() << endl;}return 0;
}

在这里插入图片描述

从输出结果来看,并不会返回空指针,并且只抛异常输出异常结果。(这里我们先不讨论异常以后会谈滴)

delete底层实现
这里我们只需要知道,delete的底层代码调用了free()函数来释放空间。

#define   free(p)               _free_dbg(p, _NORMAL_BLOCK)void operator delete(void *pUserData)
{_CrtMemBlockHeader * pHead;RTCCALLBACK(_RTC_Free_hook, (pUserData, 0));if (pUserData == NULL)return;_mlock(_HEAP_LOCK);  /* block other threads */__TRY/* get a pointer to memory block header */pHead = pHdr(pUserData);/* verify block type */_ASSERTE(_BLOCK_TYPE_IS_VALID(pHead->nBlockUse));_free_dbg( pUserData, pHead->nBlockUse );__FINALLY_munlock(_HEAP_LOCK);  /* release other threads */__END_TRY_FINALLYreturn;
}

总结:

  • new的底层实现是通过malloc,只不过多了抛异常。
  • delete的底层说明最终是通过free函数来释放空间的。

2.6 new和delete的实现原理

对于内置类型
new/delete和malloc/free基本一致,不同的是new开辟空间失败会抛异常,而malloc开辟失败返回NULL指针。

对于自定义类型
我们现在已经知道了

  • new的实现原理

    1.调用operator new 函数申请空间
    2.在申请的空间上执行构造函数

  • delete的实现原理
    1.在空间上执行析构函数,完成对象中资源的清理工作
    2.调用operator delete函数释放空间。


那么new[] 和 delete[] 的原理是什么样的?

  • new[] 的实现原理

    1.调用operator new[] 函数,实际调用operator new函数完成N个对象空间的申请,而operator new 实际也是用malloc完成空间申请。
    2.在申请的空间上执行多次构造函数

  • delete[] 的实现原理

    1.调用operator delete[] 函数,实际调用operator delete函数完成空间的释放,而operator delete 实际也是通过free完成释放。
    2.先会多次调用析构函数,完成多个对象空间的资源清理

2.7 定位new表达式的使用

new是先开空间,再调用构造函数初始化。
定位new是在已分配原始内存空间中调用构造函数初始化一个对象。

使用方式:
new(place_address)type 或者 new(place_address)type(initializer-list)
place_address是一个指针,initializer-list是类型初始化列表。

class A
{
public:A(int a = 0):_a(a){cout << "A():" << this << endl;}~A(){cout << "~A():" << this << endl;}private:int _a;
};int main()
{A* p1 = (A*)malloc(sizeof(A));if (p1 == nullptr){perror("malloc fail");exit(-1);}new(p1)A(1);p1->~A();free(p1);A* p2 = (A*)operator new(sizeof(A));new(p2)A(1);p2->~A();operator delete(p2);return 0;
}

在这里插入图片描述

定位new在实际中一般是配合内存池进行使用。因为内存池分配出的内存没有初始化,所以如果是自定义类型的对象,需要使用new的定义表达式进行显示调构造函数进行初始化。


接下来我们就可以总结一下malloc和new的区别

  1. malloc是函数,new是操作符。。
  2. malloc在开多个空间需要计算大小,new只需要填写数字就行。
  3. malloc返回值是void*需要强转,new只需要声明类型。
  4. malloc开辟失败返回空指针,new开辟失败抛异常。
  5. malloc只开空间不初始化,new即开空间又调用构造函数初始化。
  6. malloc相匹配释放空间的free只释放空间,new相匹配的delete不仅释放空间还调用析构函数。


3、初识模板

3.1 函数模板

当我们面对同一个函数需要处理多个类型的情况时,函数模板可以给我们提供很大的便利。

函数模板的格式
template<typename T1, typename T2,…,typename Tn>
返回值类型 函数名(参数列表){}

typename是用来定义模板参数关键字,也可以使用class。

用法:

template<typename T>
void Swap(T& left, T& right)
{T temp = left;left = right;right = temp;
}int main()
{int a = 1, b = 2;Swap(a, b);double c = 1.1, d = 2.2;Swap(c, d);cout << a << " " << b << endl;cout << c << " " << d << endl;return 0;
}

值得注意的是,上面Swap函数是一个模板(一个蓝图),本身并没有实例化,只有当我们调用的时候编译器会根据参数类型自动推演实例化一个函数。

我们知道编译器会根据参数类型进行推演,那么当参数类型不同会发生什么呢

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 10.1, d2 = 20.2;//自动推演//cout << Add(a1, d1) << endl; //编译报错cout << Add(a1, a2) << endl;cout << Add(d1, d2) << endl;//强转可以运行,不过强转生成临时变量具有常性,函数需要加constcout << Add((double)a1, d2) << endl;cout << Add(a1, (int)d2) << endl;//显示实例化//这种写法可以指定类型//a1 到double类型 有隐式类型转换会产生临时变量具有常性 需要加constcout << Add<double>(a1, d2) << endl;cout << Add<int>(a1, d2) << endl;return 0;
}

参数不一致第一种处理方法就是强转或者指定类型

//这样写两个不同类型就能直接相加
template<class T1, typename T2>
T1 Add(const T1& left, const T2& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 10.1, d2 = 20.2;cout << Add(a1, a2) << endl;cout << Add(d1, d2) << endl;cout << Add(a1, d2) << endl;cout << Add(d1, a2) << endl;return 0;
}

参数不一致第二种处理方法就是添加两个类型


当我们写了的函数能用的话,编译器也不会再生成,只有指定调通用的才会推演生成。

int Add(int left, int right)
{return left + right;
}template<class T>
T Add(T left, T right)
{return left + right;
}int main()
{int a = 1, b = 2;Add(a, b);//调专门的不调通用的Add<int>(a, b); //要调通用的编译器自己推演return 0;
}


3.2 类模板

类模板的定义格式:
template<class T1, class T2, …, class Tn>
class 类模板名
{
// 类内成员定义
};

例子:

template<typename T>
class Stack
{
public:Stack(int capacity = 4){cout << "Stack(int capacity = )" <<capacity<<endl;_a = (T*)malloc(sizeof(T)*capacity);if (_a == nullptr){perror("malloc fail");exit(-1);}_top = 0;_capacity = capacity;}~Stack(){cout << "~Stack()" << endl;free(_a);_a = nullptr;_top = _capacity = 0;}void Push(const T& x){// ....// 扩容_a[_top++] = x;}private:T* _a;int _top;int _capacity;
};int main()
{//类型是Stack<double> st1使得类实例化Stack<double> st1;st1.Push(1.1);//类型是Stack<int> st2使得类实例化Stack<int> st2;st2.Push(1);return 0;
}

类模板不是真正的类,只有当实例化后才是类。
由于类的实例化一开始不会传参,所以类模板没有推演时机。
虽然是同一个类模板实力化的,但是参数不同,类型不同,大小不一样。

并且类模板中成员函数的定义与声明最好写在同一文件中

如果定义与声明分离,由于类模板未实例化问题,会发生链接错误。
如果一定要声明与定义分离,需要在每个包含类模板的文件实例化(这就很麻烦,所以尽量避免声明与定义分离)。

头文件中实例化写法

//stack<int>类实例化
template
class stack<int>;

本章完~

http://www.dtcms.com/wzjs/188982.html

相关文章:

  • 网站建设卖给别人可以吗网络服务器图片
  • 上海网站开发培训网站建设与营销经验
  • 送给做网站的锦旗语网站排名分析
  • 公众号网站怎么做的企业网站的主要类型有
  • 辽宁省住建厅建设网站营销方案模板
  • 搬瓦工服务器用来做网站建站平台如何隐藏技术支持
  • 如何配置 网站二级域名短视频营销策划方案
  • 重庆企业的网站建设宁波seo快速优化课程
  • crm软件系统的构成包括seo搜索引擎优化费用
  • 做虾网站该起啥名好网站优化推广平台
  • 西安外贸网站开发如何推广app赚钱
  • 网站要素百度竞价推广的技巧
  • 洛阳数码大厦做网站的在几楼历史权重查询
  • uc网站怎么做网络营销策划书案例
  • 社保网站做员工用工备案吗整站排名优化公司
  • 自己如何创建网站快速提高排名
  • 网站建设合同 完整版厦门排名推广
  • 网站建设费会计科目我是做推广的怎么找客户
  • dreamweaver8可以做资源下载网站网站可以自己建立吗
  • 做网站用笔记本做服务器吗seo官网
  • 网站开发进修优化课程设置
  • 企业做网站需要多少钱优秀软文范例200字
  • 做渐变色的网站最新热搜新闻事件
  • 网站策划书案例展示磁力猫引擎
  • 美妆网站建设规划app广告投放价格表
  • 商业网站怎么做网站推广的全过程
  • 网站建设评审会的通知友情链接大全
  • 成都正规搜索引擎优化快速将网站seo
  • 林州网站建设熊掌号天天seo百度点击器
  • 做网站好看的旅行背景图片最新热搜新闻