C++ 内存管理和模板与STL

        此篇目是之后各种C++库的基础

目录

内存管理

        内存分布

        内存管理方式 

        new和delete 

         operator new 与 operator delete函数

        实现原理 

         定位new表达式(placement-new)

 模板基础

        泛型编程

        模板

        函数模板

        类模板

STL

         组成部分


内存管理

        内存分布

int globalVar = 1; //全局变量 静态区
static int staticGlobalVar = 1;//静态变量 静态区
void Test()
{
static int staticVar = 1;//静态变量 静态区
int localVar = 1;//局部变量 栈
int num1[10] = { 1, 2, 3, 4 };//局部变量 栈
char char2[] = "abcd";//局部变量 栈
const char* pChar3 = "abcd";//指针 栈 但是"abcd"是代码段(常量区)
int* ptr1 = (int*)malloc(sizeof(int) * 4);//指针 栈 指向堆
int* ptr2 = (int*)calloc(4, sizeof(int));//指针 栈 指向堆
int* ptr3 = (int*)realloc(ptr2, sizeof(int) * 4);//指针 栈 指向堆
free(ptr1);
free(ptr3);
}

栈                存储非静态局部变量/函数参数/返回值 向下增长

内存映射段 高效的I/O映射方式,用于装载一个共享的动态内存库

堆                程序运行时动态内存分配,向上生长

数据段         全局数据与静态数据

代码段         可执行的代码/只读的常量

        内存管理方式 

        内存管理方式有 malloc/calloc/realloc/free

  1. malloc

    • 内存来源:在堆区申请一块连续的指定大小的内存块区域。
    • 初始化:不会进行内存初始化,即分配的内存区域可能包含任意数据。
  2. calloc

    • 内存来源:与malloc类似,也是用于动态地分配内存空间。
    • 初始化:会在分配内存时自动将内存清零,即将分配的内存区域的每一位都初始化为零。因此,calloc也被称为“零初始化内存分配器”。
  3. realloc

    • 内存来源:用于重新分配之前通过malloc、calloc或realloc分配的内存块。
    • 初始化:realloc本身不会进行初始化操作,但它会调整已分配内存块的大小,并根据需要调整内存块的位置。如果重新分配的内存区域比原来的大,则新分配的部分不会进行初始化。

        但是在C++中 引入了新的内存管理方式

        new和delete 

        因为有一些方面 原始的申请会有些无力 因此C++推出了自己的申请函数

int main()
{int* ptr1 = new int;//申请一个int空间int* ptr2 = new int(10);//申请一个int空间 并初始化为10int* ptr3 = new int[3];//申请3个int空间 可以认为是 数组delete ptr1;//普通的就用delete删除delete ptr2;delete[] ptr3; //[]出来的要用[]删除
}

        new和delete在对自定义类型操作时 也会有不同的变化

对自定义类型操作时

        new会调用自定义类型的构造函数

        delete会调用自定义类型的析构函数

malloc和free不会去调用

class A
{
public:A(int num = 0) :a(num) { cout << "A()" << endl;}~A() { cout << "~A()" << endl;}
private:int a;
};int main()
{A* ptra = new A(1);delete ptra;A* ptrb = new A[5];delete[] ptrb;return 0;
}

 

         operator new 与 operator delete函数

new与delete                              是用户进行内存申请和释放的 操作符

operator new与operator delete 是系统提供的                           全局函数

new    在底层 调用 operator new

delete 在底层 调用 operator delete

operator new    是通过 malloc 申请空间

operator delete 是通过 free     释放空间

        实现原理 

        new原理

        调用operator new申请空间 然后用构造函数完成对象构造

        delete原理

        调用析构函数清理对象 然后调用operator delete函数释放对象空间 

        new T[N]原理

        调用operator new[]函数,在operator new[]中调用operator new函数完成对N个对象的空间申请 然后用N此构造函数

        delete[]原理

        在要释放的对象空间上完成N次析构函数,然后对N个对象中资源的清理,然后调用operator delete[]释放空间,也就是在operator 

         定位new表达式(placement-new)

        解释

        定位new表达式是在已经分配的原始内存空间中调用构造含初始化了一个对象

        使用格式

        new (place_address)type 或者 new(place_address) type(initializer-list)

        place_address必须是一个指针initializer-list是类型的初始化列表 

         使用场景

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

class A
{
public:A(int num = 0) :a(num) { cout << "A()"<< a;}~A() { cout << "~A()";}
private:int a;
};int main()
{A* p1 = (A*)malloc(sizeof(A));new(p1)A;//new重定位p1->~A();free(p1);cout << endl;A* p2 = (A*)malloc(sizeof(A));new(p2)A(10);//new重定位 然后给初始值p2->~A();free(p2);
}

 模板基础

        泛型编程

        用函数重载实现交换函数

void Swap(int& left, int& right)
{
int temp = left;
left = right;
right = temp;
} 
void Swap(double& left, double& right)
{
double temp = left;
left = right;
right = temp;
} 
void Swap(char& left, char& right)
{
char temp = left;
left = right;
right = temp;
}

        从例子中看到,重载的函数仅仅只是类型不同,复用成度并不高,如果出现新的类型,就需要用户自己新增,而且无法一起修正错误,往往伴随着,错一个就错一堆,那么就要用到"模板" 了

        模板

        C++中存在一个模具,根据模具中填充的不同类型,获得不同的代码,泛型编程解释编译与类型无关的通用代码,方便代码复用.

模板分 函数模板 和 类模板

        函数模板

        概念

        函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生的函数的特定类型的版本

          格式

templat<typename T1,typename T2....>

返回值 函数名 参数

template<typename T1, class T2>
void Swap(T1& left, T2& right)
{T1 temp = left;left = right;right = temp;
}

        原理

        通过编译器使用的特殊方式产生具体类型函数的模具,模板就是将本应该我们做的重复的事情交给了编译器 

        实例化

        隐式调用

template<typename T1, class T2>
void Swap(T1& left, T2& right)
{T1 temp = left;left = right;right = temp;
}
int main()
{int a = 1;double b = 2;Swap(a, b);cout << a << b << endl;return 0;
}

        显式调用

template<typename T1, class T2>
void Swap(T1& left, T2& right)
{T1 temp = left;left = right;right = temp;
}
int main()
{int a = 1;double b = 2;Swap<int,double>(a, b);cout << a << b << endl;return 0;
}

        模板参数匹配原则

        我称之为:最多匹配优先原则,就是匹配拟合度最高的优先被调用

        而且无论是已存在的还是模板,都是按照拟合度最高的匹配 

void Swap(int left, int right)//优先匹配 int int
{int temp = left;left = right;right = temp;
}template<typename T1>//匹配 两个相同参数
void Swap(T1& left, T1& right)
{T1 temp = left;left = right;right = temp;
}template<typename T1,int>//匹配 T1 和 int 优先
void Swap(T1& left, int& right)
{T1 temp = left;left = right;right = temp;
}template<typename T1, class T2>//任意两个类型
void Swap(T1& left, T2& right)
{T1 temp = left;left = right;right = temp;
}

        类模板

        也就是 使用模板参数当作类里面参数 

             定义格式

template<class T1,class T2>
class name
{
public://...
private:T1 _t1;T2 _t2://...
};

        注意

        模版不建议声明和定义分离到两个文件.h 和.cpp会出现链接错误

        实例化

template<class T1,class T2>
class name
{
public://...
private:T1 _t1;T2 _t2://...
};
int main()
{name<int, int> A;name<int, char> B;
}

STL

        STL是C++标准库的重要组成部分,是标准模板库,不仅是一个可复用的组件库,而且是一个包含数据机构与算法的软件框架

         组成部分

        仿函数 算法 迭代器 空间配置器 容器 配接器 

        本章结言

我将会在下一章开始写 

string

vector

list

stack

queue

等基础STL的应用与实现

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/64032.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

在 Ubuntu 24.04.1 LTS (WSL) 中使用 openssl 生成 keybox.xml

看到“生成 keybox.xml”&#xff0c;大概率都会联想到 PIF 和 Tricky Store。这里就不多解释它们的用途了。最近在网上看到生成非 AOSP keybox 的教程&#xff0c;在这里做一些补充&#xff0c;并将代码打包成一个 Python 脚本。 参考自&#xff1a; Idea 提供者&#xff1a…

Redis篇-1--入门介绍

1、Redis概述 ‌Redis&#xff08;Remote Dictionary Server&#xff09;&#xff0c;全称为远程字典服务。‌是一个开源的、使用C语言编写的、支持网络交互的、可基于内存也可持久化的Key-Value数据库。 Redis提供了多种数据类型的存储&#xff0c;来适应不同场景下的存储需…

antdv-<a-button>中属性的使用

UI组件库&#xff08;User Interface Component Library&#xff09;是一种预先构建好的、可重用的用户界面元素集合&#xff0c;旨在帮助开发者更快速、更简便地构建用户界面。这些组件通常包括按钮、表单、导航栏、模态框等&#xff0c;能够提供一致的外观和交互风格&#xf…

简单的多网卡选择指定网卡ip注册

简单的多网卡选择指定网卡ip注册 我们公司服务器上面有多个网卡&#xff0c;多网卡则本地ip有多个ip,我们启动服务的时候需要选定他特定的ip&#xff0c;我们服务需要特定的ip进行注册&#xff0c;才能进行正常的通讯功能&#xff0c;我们需要使用如下配置进行特定ip选择&…

鸿蒙NEXT开发案例:颜文字搜索器

【引言】 本文将介绍一个名为“颜文字搜索器”的开发案例&#xff0c;该应用是基于鸿蒙NEXT平台构建的&#xff0c;旨在帮助用户快速查找和使用各种风格的表情符号。通过本案例的学习&#xff0c;读者可以了解如何在鸿蒙平台上进行数据处理、UI设计以及交互逻辑的实现。 【环…

快速部署一套K8s集群-v1.28

快速部署一套K8s集群-v1.28 1.前置知识点 1.1 生产环境可部署Kubernetes集群的两种方式 目前生产部署Kubernetes集群主要有两种方式: kubeadmKubeadm是一个K8s部署工具,提供kubeadm init和kubeadm join,用于快速部署Kubernetes集群。 二进制包从github下载发行版的二进…

【硬件测试】基于FPGA的4ASK调制解调通信系统开发与硬件片内测试,包含信道模块,误码统计模块,可设置SNR

目录 1.算法仿真效果 2.算法涉及理论知识概要 3.Verilog核心程序 4.开发板使用说明和如何移植不同的开发板 5.完整算法代码文件获得 1.算法仿真效果 本文是之前写的文章: 《基于FPGA的4ASK调制解调系统,包含testbench,高斯信道模块,误码率统计模块,可以设置不同SNR》 的…

用友U8+ API接口使用教程

前言 U8和其他的公开的开放API接口有一些差异&#xff0c;他是需要先对接的到代理服务器&#xff0c;通过代理服务器进行对接&#xff0c;所以只要保证U8能上网就能对接&#xff0c;和畅捷通T的模式有点类似 流程&#xff1a; 注册成为开发者&#xff08;用于创建用友U8 API应…

检查读取数据寄存器输出的多扇出

为使第二寄存器被 RAM 原语吸收&#xff0c;来自存储器阵列的数据输出位的扇出必须为 1 。这在下图中进行了说明。 检查地址 / 读取数据寄存器上的复位信号 不应复位存储器阵列。只有 RAM 的输出可以容许复位。复位必须是同步的&#xff0c;以便将输出寄存器推断到 RAM 基元…

MongoDB-ObjectID 生成器

前言 MongoDB中一个非常关键的概念就是 ObjectID&#xff0c;它是 MongoDB 中每个文档的默认唯一标识符。了解 ObjectID 的生成机制不仅有助于开发人员优化数据库性能&#xff0c;还能帮助更好地理解 MongoDB 的设计理念。 什么是 MongoDB ObjectID&#xff1f; 在 MongoDB …

渗透测试工具 -- SQLmap安装教程及使用

随着网络安全问题日益严峻&#xff0c;渗透测试成为了保护信息安全的重要手段。而在渗透测试的众多工具中&#xff0c;SQLmap凭借其强大的自动化SQL注入检测和利用能力&#xff0c;成为了网络安全专家必备的利器。那么&#xff0c;你知道如何高效地使用SQLmap进行漏洞扫描吗&am…

SpringBoot 整合 RabbitMQ 实现流量消峰

RabbitMQ 即一个消息队列&#xff0c;主要是用来实现应用程序的异步和解耦&#xff0c;同时也能起到消息缓冲&#xff0c;消息分发的作用。 消息中间件在互联网公司的使用中越来越多&#xff0c;刚才还看到新闻阿里将 RocketMQ 捐献给了 Apache&#xff0c;当然了今天的主角还…

Git-分支(branch)常用命令

分支 我们在做项目开发的时候&#xff0c;无论是软件项目还是其他机械工程项目&#xff0c;我们为了提高效率以及合理的节省时间等等原因&#xff0c;现在都不再是线性进行&#xff0c;而是将一个项目抽离出诸进行线&#xff0c;每一条线在git中我们就叫做分支&#xff0c;bran…

springboot425滑雪场管理系统(论文+源码)_kaic

摘要 近年来&#xff0c;信息化管理行业的不断兴起&#xff0c;使得人们的日常生活越来越离不开计算机和互联网技术。首先&#xff0c;根据收集到的用户需求分析&#xff0c;对设计系统有一个初步的认识与了解&#xff0c;确定滑雪场管理系统的总体功能模块。然后&#xff0c;详…

好玩的汇编编译器NASM:一款基于x86架构的汇编与反汇编软件

好玩的汇编编译器NASM This is the project webpage for the Netwide Assembler (NASM), an asssembler for the x86 CPU architecture portable to nearly every modern platform, and with code generation for many platforms old and new. Netwide Assembler&#xff08;…

【AI知识】有监督学习分类任务之支持向量机

1.支持向量机概念 支持向量机&#xff08;Support Vector Machine, SVM&#xff09; 是一种有监督学习算法&#xff0c;主要用于分类任务&#xff08;也可用于回归任务&#xff0c;即支持向量回归&#xff0c;SVR&#xff09;。SVM的核心思想是找到一个最优的超平面&#xff0…

杭州乘云联合信通院发布《云计算智能化可观测性能力成熟度模型》

原文地址&#xff1a;杭州乘云联合中国信通院等单位正式发布《云计算智能化可观测性能力成熟度模型》标准 2024年12月3日&#xff0c;由全球数字经济大会组委会主办、中国信通院承办的 2024全球数字经济大会 云AI计算创新发展大会&#xff08;2024 Cloud AI Compute Ignite&…

【经典】制造供应链四类策略(MTS、MTO、ATO、ETO)细说

关注作者 制造供应链的牛鞭问题与复杂问题主要是从两个方面解决&#xff0c;一是同步化供应链消减从需求到供应的放大效应&#xff0c;二是供应链细分&#xff0c;针对不同的客户、不同的需求供应的匹配策略来应对复杂性&#xff0c;更好的满足客户并以最低的总成本来实现。 对…

实时日志与发展:Elasticsearch 推出全新专用的 logsdb 索引模式

作者&#xff1a;来自 Elastic Mark Settle, George Kobar 及 Amena Siddiqi Elastic 最新发布的 logsdb 索引模式是专为日志管理优化的功能&#xff0c;旨在提升日志数据的存储效率、查询性能以及整体可用性。这个模式专注于满足现代日志处理需求&#xff0c;提供更高效的日志…

React Image Crop——在React应用中轻松实现图片裁剪功能

React Image Crop是一个用于在React应用程序中裁剪和调整图像的库。它提供了一个简单而强大的界面&#xff0c;允许用户选择和调整裁剪区域&#xff0c;并生成裁剪后的图像。 什么是React Image Crop&#xff1f; React Image Crop是一个开源的React组件&#xff0c;用于在浏览…