C++:内部类,匿名对象,操作符new与delete

news2024/12/28 19:45:51

一.内部类

1.如果一个类定义在另一个类的内部,这个内部类就叫做内部类。内部类是一个独立的类,跟定义在全局相比,他只是受外部类类域限制和访问限定符限制,所以外部类定义的对象中不包含内部类。
2.内部类默认是外部类的友元类。
3.内部类本质也是一种封装,当A类跟B类紧密关联,A类实现出来主要就是给B类使用,那么可以考虑把A类设计为B的内部类,如果放到private/protected位置,那么A类就是B类的专属内部类,其他地方都用不了。 

比如下面这个例子:

这两种写法效果是一样的,B此时就是A的内部类。

二.匿名对象

匿名对象当下的阶段可能没有太大的作用,但在模板之后便会有显著作用了,它的特点如下:

1.用类型(实参) 定义出来的对象叫做匿名对象,相比之前我们定义的 类型 对象名(实参) 定义出来的叫有名对象。
2.匿名对象生命周期只在当前一行,一般临时定义一个对象当前用一下即可,就可以定义匿名对象。 

牛客网上的一道题如下:日期累加_牛客题霸_牛客网 (nowcoder.com)

这里就可以使用下匿名对象来解决这道题(虽然效果不是太明显了)

#include <iostream>
#include <assert.h>
using namespace std;

class Date {
  public:
    Date()
        : _year(2024),
          _month(9),
          _day(13) {
        int m = 0;
        int n = 0;
        cin >> m;
        while (m--) {
            cin >> _year >> _month >> _day >> n;
            *this += n;
            DatePrint(*this);
        }
    }
    int GetMonthDay() {
        assert(_month <= 12 || _month > 0);
        static int arr[13] = { -1, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
        if (_month == 2 && ((_year % 4 == 0 && _year % 100 != 0) || _year % 400 == 0)) {
            return 29;
        }
        return arr[_month];
    }
    Date& operator-=(int i) {
        if (i < 0) {
            return *this += -i;
        }
        _day -= i;
        while (_day < 0) {
            _day += GetMonthDay();
            --_month;
            if (_month == 0) {
                --_year;
                _month = 12;
            }
        }
        return *this;
    }
    Date& operator+=(int i) {
        if (i < 0) {
            return *this -= -i;
        }
        _day += i;
        while (_day > GetMonthDay()) {
            _day -= GetMonthDay();
            ++_month;
            if (_month == 13) {
                ++_year;
                _month = 1;
            }
        }
        return *this;
    }
    void DatePrint(const Date& d) {
        if(d._month >= 10 && d._day >= 10)
        cout << d._year << "-" << d._month << "-" << d._day << endl;
        else if(d._month < 10 && d._day >= 10)
        cout << d._year << "-" << 0 << d._month << "-" << d._day << endl;
        else if(d._month == 10 && d._day < 10)
        cout << d._year << "-" << d._month << "-" << 0 << d._day << endl;
        else if(d._month < 10 && d._day == 10)
        cout << d._year << "-" << 0 << d._month << "-" << d._day << endl;
        else
        cout << d._year << "-" << 0 << d._month << "-" << 0 << d._day << endl;;
    }

  private:
    int _year;
    int _month;
    int _day;
};
int main() {
    Date();
    return 0;
}

我们可以直接通过匿名对象来调用其构造函数,直接完成所有函数的调用来实现这道题目(写的有些糅杂还请见谅)。

三.C/C++内存管理

3.1new与delete

我们在学习C的时候已经知道,通过使用malloc/realloc/calloc/free函数来管理我们开辟的动态内存。在C++中我们将介绍两个新的操作符new和delete来进行动态内存管理基本使用方式如下:

void Test()
{
   // 动态申请一个int类型的空间
   int* ptr4 = new int;
   // 动态申请一个int类型的空间并初始化为10
   int* ptr5 = new int(10);
   // 动态申请10个int类型的空间
   int* ptr6 = new int[3];
   delete ptr4;
   delete ptr5;
   delete[] ptr6;
}

申请和释放单个元素的空间,使用new和delete操作符,申请和释放连续的空间,使用
new[]和delete[],new与delete在处理内置类型时,与malloc/free别无差异,当开辟的是自定义类型时,new与delete会在开辟的时候调用自定义类的构造和析构函数,注意匹配起来使用,否则会出现未定义行为,比如我们用下面这个例子来解释:

char* p = new char[100];
delete p;

当你使用new[]来动态分配内存,如果你使用错误的delete(而不是delete[])来释放这段内存(如delete p),虽然它可能不会立即出现问题,但实际上是未定义行为。 

在使用delete p来释放使用new[]分配的内存时,C++ 标准明确规定这是未定义行为。然而在实践中,对于简单类型如charint等原始数据类型,由于它们没有复杂的构造函数和析构函数,delete没有明显的副作用。系统仅仅会释放内存空间,这与free的行为类似。

所以在使用内置类型这种简单的类型时,一般不会出现问题。一旦使用自定义类型时,编译器就会因无法正确处理所有对象的析构和内存释放而导致报错。

3.2operator new与operator delete 

new和delete是用户进行动态内存申请和释放的操作符,operator new 和operator delete是
系统提供的全局函数,new在底层调用operator new全局函数来申请空间,delete在底层通过
operator delete全局函数来释放空间。

/*
operator new:该函数实际通过malloc来申请空间,当malloc申请空间成功时直接返回;申请空间
失败,尝试执行空 间不足应对措施,如果改应对措施用户设置了,则继续申请,否
则抛异常。
*/
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); /* 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_FINALLY
		return;
}

通过上述两个全局函数的实现知道,operator new实际也是通过malloc来申请空间,如果
malloc申请空间成功就直接返回,否则执行用户提供的空间不足应对措施,如果用户提供该措施
就继续申请,否则就抛异常。operator delete最终是通过free来释放空间的。 

3.3new和delete的实现原理 

3.3.1内置类型 

如果申请的是内置类型的空间,new和malloc,delete和free基本类似,不同的地方是:
new/delete申请和释放的是单个元素的空间,new[]和delete[]申请的是连续空间,而且new在申
请空间失败时会抛异常,malloc会返回NULL。

3.3.2自定义类型 

(1)new的原理
1. 调用operator new函数申请空间
2. 在申请的空间上执行构造函数,完成对象的构造 

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

(3)new T[N]的原理
1. 调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对
象空间的申请
2. 在申请的空间上执行N次构造函数

(4)delete[]的原理
1. 在释放的对象空间上执行N次析构函数,完成N个对象中资源的清理
2. 调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释
放空间

3.4malloc/free和new/delete的区别

malloc/free和new/delete的共同点是:都是从堆上申请空间,并且需要用户手动释放。不同的地
方是:
1. malloc和free是函数,new和delete是操作符
2. malloc申请的空间不会初始化,new可以初始化
3. malloc申请空间时,需要手动计算空间大小并传递,new只需在其后跟上空间的类型即可,
如果是多个对象,[]中指定对象个数即可
4. malloc的返回值为void*, 在使用时必须强转,new不需要,因为new后跟的是空间的类型
5. malloc申请空间失败时,返回的是NULL,因此使用时必须判空,new不需要,但是new需
要捕获异常
6. 申请自定义类型对象时,malloc/free只会开辟空间,不会调用构造函数与析构函数,而new
在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成
空间中资源的清理释放

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

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

相关文章

基于Java+Mysql实现(web)大型企业管理系统

技术报告 第一章 系统概述 包括用户管理、权限管理、软件项目管理、软件模块管理、测试用例管理、测试任务分配、bug管理等功能。实现公司不同部门间团队协作&#xff0c;管理人员也能够更加有效的把控系统开发的进度。 本实验综合应用JavaWeb编程中的Servlet&#xff0c;JS…

【渗透测试】-CVE-2016-4437-Shiro550漏洞复现

Shiro550漏洞爆出的时间是2016年为第4437个漏洞&#xff0c;所以它的CVE编码是2016-4437 文章目录 前言 什么是Shiro550漏洞&#xff1f; 1.Shiro550漏洞原理&#xff1a; 2.漏洞利用 3.漏洞复现&#xff1a; 前提&#xff1a;下载并打开vulhub靶场。 CVE-2016-4437-shiro550漏…

CentOS 7停更官方yum源无法使用,更换阿里源

CentOS 7官方源已经停止维护&#xff0c;导致无法使用yum更新软件。通过尝试使用阿里云、清华大学等第三方源解决&#xff0c;现以阿里云第三方源进行配置&#xff1a; 1、备份原有的yum源配置文件 # cp -a /etc/yum.repos.d /etc/yum.repos.d.bak 2、删除原有的yum源配置文…

什么是交换机级联?

在现代计算机网络中&#xff0c;交换机级联是一种广泛应用的技术&#xff0c;有助于提升网络的扩展性和灵活性。本文将深入探讨交换机级联相关知识&#xff0c;详细介绍其基本概念和连接配置方法&#xff0c;并对常见技术问题进行解答。 交换机级联概述 交换机级联是指通过将…

windows server2012 配制nginx安装为服务的时候,直接跳要安装.net框架,用自动的安装,直接失败的解决。

1、上一个已成功在安装过程中的图&#xff1a; 2、之前安装过程中错误的图&#xff1a; 3、离线安装解决&#xff1a; 下载.net framework 3.5&#xff0c;然后解压后&#xff0c;选择指定备用源路径&#xff0c;然后选择.net安装包所在目录&#xff1a; 只要指定上面全路径就…

WebGL系列教程八(GLSL着色器基础语法)

目录 1 前言2 基本原则3 基本数据类型4 顶点着色器和片元着色器4.1 声明4.2 初始化项目4.3 赋值 5 结构体5.1 声明5.2 赋值 6 函数6.1 基本结构6.2 自定义函数6.3 常用内置函数 7 精度8 其他9 总结 1 前言 通过前七讲&#xff0c;我们已经见过了WebGL中的部分基础语法&#xff…

初始爬虫1(补充)

TCP 和 UDP 是什么&#xff1f; TCP&#xff08;Transmission Control Protocol&#xff09;和 UDP&#xff08;User Datagram Protocol&#xff09;都是传输层协议&#xff0c;它们负责在计算机网络上发送和接收数据包。两者有不同的特性和适用场景&#xff1a; TCP&#xff0…

文档团队如何组成?

经常有朋友问我文档团队是由怎样背景的人组成的&#xff1f;今天来聊聊这个话题。 中国贸促会和技术传播专委会以及lnfomagic学院2023年在技术传播行业做了一个调查&#xff0c;在收到的231份有效样本中显示&#xff0c; 受访群体的背景主要是两大类&#xff0c;他们分别是技术…

视频推镜拍摄SDK解决方案,创新短视频玩法

在当今社交媒体盛行的时代&#xff0c;短视频已成为人们分享生活、展示创意的重要方式。美摄科技推出的视频推镜拍摄SDK解决方案&#xff0c;为用户提供了一种全新的短视频创作体验。 一、什么是视频推镜拍摄SDK解决方案&#xff1f; 美摄科技的视频推镜拍摄SDK解决方案是一种…

虹科方案 | 精准零部件测试!多路汽车开关按键功能检测系统

欢迎关注虹科&#xff0c;为您提供最新资讯&#xff01; #LIN/CAN总线 #零部件测试 #CAN数据 导读 在汽车制造业中&#xff0c;零部件的安全性、功能性和可靠性是确保车辆整体性能的关键。虹科针对车辆零部件的LIN/CAN总线仿真测试&#xff0c;提出了基于虹科Baby-LIN系列产…

详细分析linux中的MySql跳过密码验证以及Bug(图文)

目录 1.问题所示2. 基本知识3. 解决方法3.1 跳过验证Bug3.2 设定初始密码 1.问题所示 发现密码验证错误&#xff0c;遗失密码 2. 基本知识 停止MySQL服务&#xff1a;sudo systemctl stop mysql 以跳过权限表模式启动MySQL&#xff1a;sudo mysqld_safe --skip-grant-tables …

移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——12.二叉树(习题)

1.根据二叉树创建字符串 . - 力扣&#xff08;LeetCode&#xff09; 我的思路&#xff1a; 1. 根节点单独讨论&#xff0c;因为&#xff0c;根节点之后才会开始有括号 2.根节点的左孩子和右孩子分别进入operation函数 operation函数&#xff1a; 1.如果root不为空&#xff0c;…

计算机毕业设计选题推荐-在线投票系统-Java/Python项目实战

✨作者主页&#xff1a;IT研究室✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Python…

程序员的 AI 启蒙:ChatGPT+ Copilot开发Vue3 仿简书项目 90%代码AI生成

在人工智能技术日益成熟的今天&#xff0c;程序员们正在迎来一场全新的编程革命。ChatGPT和Copilot等AI工具的出现&#xff0c;让代码生成不再是遥不可及的梦想。本文将带你体验如何利用这些先进技术&#xff0c;仅用90%的代码量&#xff0c;开发出一个基于Vue3的仿简书项目&am…

电容笔有必要买吗?西圣、wiwu、倍思好不好用?王者产品测评对比

当下很多年轻人都热衷于使用 iPad 来进行学习与娱乐&#xff0c;因此电容笔还是很有必要买的&#xff0c;它可以提升我们的效率。然而作为一名数码测评博主&#xff0c;我十分清楚一旦选到质量不佳的产品&#xff0c;就会出现诸如断触、延迟等问题&#xff0c;进而拉低效率。 …

2024年9月一区SCI-神经种群动态优化算法NPDOA-附Matlab免费代码

引言 本期介绍了一种受脑神经科学启发的元启发式算法&#xff0c;称为神经种群动态优化算法Neural population dynamics optimization algorithm(NPDOA)的元启发式算法。该成果于2024年9月最新发表在中科院1区 Top SCI期刊 Knowledge-Based Systems。 原文作者将NPDOA与其他9种…

大学生必备10个AI工具网站,辅助完成辩论/开题/实践/形势政策报告、创新创业计划书、思想汇报、心得感悟等作业,提升学习效率和学术表现!

大学新生和学长学姐们都已经开学了&#xff0c;忙碌的课程和多样的作业也随之开始&#xff0c;下面将给大学生们介绍10个辅助完成作业、寻找灵感&#xff0c;提升学习专注力和学术表现的AI工具~ 1、笔墨写作 笔墨写作 - 领先的写作智能AI创作平台 | 官方首页笔墨写作是一款专…

深度剖析去中心化存储:IPFS、Arweave 和 BNB Greenfield 的技术革新与生态系统演进

引言&#xff1a; 在数字时代的浪潮中&#xff0c;数据已然成为驱动创新和决策的核心资产。然而&#xff0c;随着数据量呈指数级增长&#xff0c;传统中心化存储模式面临着前所未有的挑战。安全漏洞、隐私泄露、数据垄断等问题日益凸显&#xff0c;促使技术界重新思考数据存储…

QT多线程编程(基础概念以及示例)

QT多线程编程 前言&#xff1a;基础夯实&#xff1a;一:多线程概述二:QT多线程的使用1. 继承QThread类2. 继承QObject类3. QtConcurrent模块 三:线程同步与通信四:线程安全五:线程管理六:总结 效果展示&#xff1a;实现功能&#xff1a;核心代码&#xff1a;mainwindow.hmythre…

2024数学建模国赛官方评阅标准发布!

​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑​↑…