C++友元与动态内存
一、友元
友元是一种定义在类外部的普通函数或类,但它需要在类体内进行说明,为了与该类的成员函数加以区别,在说明时前面加以关键字friend。友元不是成员函数,但是它可以访问类中的私有成员。
类具有封装和信息隐藏的特性。只有类的成员函数才能访问类的私有成员,程序中的其他函数是无法访问私有成员的。非成员函数可以访问类中的公有成员,但是如果将数据成员都定义为公有的,这又破坏了隐藏的特性。另外,应该看到在某些情况下,特别是在对某些成员函数多次调用时,由于参数传递,类型检查和安全性检查等都需要时间开销,而影响程序的运行效率。
目 的:提高程序的运行效率
缺 点:破坏了类的封装性和隐藏性
1、友元函数
类中私有和保护的成员在类外不能被访问。友元函数是一种定义在类外部的普通函数,其特点是能够访问类中私有成员和保护成员,即类的访问权限的限制对其不起作用。
友元函数需要在类体内进行说明,在前面加上关键字friend。
一般格式为:
friend <type> FuncName(<args>);
友元函数不是成员函数,用法也与普通的函数完全一致,只不过它能访问类中所有的数据。友元函数破坏了类的封装性和隐蔽性,使得非成员函数可以访问类的私有成员。
一个类的友元可以自由地用该类中的所有成员。
示例:
#include <iostream>
using namespace std;class CTestFriend
{
public:CTestFriend() // 默认的构造函数{cout << "调用默认的构造函数CTestFriend::CTestFriend().\n";}CTestFriend(int a, int b){x = a;y = b;}int getx() {return x;}int gety(){return y;}int sum(){return x + y;}friend int sum(CTestFriend &obj);private:int x, y;
};int sum(CTestFriend &obj)
{return obj.x + obj.y;
}int sumxy(CTestFriend &obj)
{return obj.getx() + obj.gety();
}int main()
{CTestFriend obj1(10, 20), obj2(100, 200), obj3(1000, 2000);cout << "\nobj1.sum():" << obj1.sum() << endl << endl;cout << "\nobj2.sum():" << sum(obj2) << endl << endl;cout << "\nobj3.sumxy():" << sumxy(obj3) << endl << endl;return 0;
}
有关友元函数的使用,说明如下:
- 友元函数不是类的成员函数
- 友元函数近似于普通的函数,它不带有this指针,因此必须将对象名或对象的引用作为友元函数的参数,这样才能访问到对象的成员。
友元函数与一般函数的不同点在于:
1、友元函数必须在类的定义中说明,其函数体可在类内定义,也可在类外定义;
2、它可以访问该类中的所有成员(公有的、私有的和保护的),而一般函数只能访问类中的公有成员。
示例:
#include <iostream>
using namespace std;class CTestFriend
{
public:CTestFriend() // 默认的构造函数{cout << "调用默认的构造函数CTestFriend::CTestFriend().\n";}CTestFriend(int a, int b) // 带有参数的构造函数{x = a;y = b;cout << "x=" << x << "," << "y=" << y << endl;}int mulxy() // 普通成员函数{return x * y;}friend int sumxy(CTestFriend &obj); // 友元函数 访问私有成员和保护成员private:int x, y;
};int sumxy(CTestFriend &obj)
{return obj.x + obj.y;
}int main()
{CTestFriend obja;CTestFriend objb(50, 100);cout << "\n两个数字之积为:" << objb.mulxy() << endl << endl;cout << "\n两个数字之和为:" << sumxy(objb) << endl << endl;return 0;
}
友元函数不受类中访问权限关键字的限制,可以把它放在类的私有部分,放在类的公有部分或放在类的保护部分,其作用都是一样的。换言之,在类中对友元函数指定访问权限是不起作用的。
友元函数的作用域与一般函数的作用域相同。谨慎使用友元函数。通常使用友元函数来取对象中的数据成员值,而不修改对象中的成员值,则肯定是安全的。
2、友元类
友元除了函数以外,还可以是类,即一个类可以作另一个类的友元。当一个类作为另一个类的友元时,这就意味着这个类的所有成员函数都是另一个类的友元函数,都可以访问另一个类中的隐藏信息(包括私有成员和保护成员)。
定义友元类的语句格式如下:friend class 类名(即友元类的类名);
示例:
#include <iostream>
using namespace std;class CTestFClassA
{
public:CTestFClassA(double a, double b) {x = a;y = b;}double getx() {return x;}double gety() {return y;}friend class CTestFClassB; // 定义类CTestFClassB为类CTestFClassA友元
private:double x, y;
};class CTestFClassB
{
public:CTestFClassB(int n = 1){k = n;}void disp(CTestFClassA &obj) {cout <<"结果为:"<< obj.x + obj.y + k;} // 求类CTestFClassA的某个数据成员到这边来操作。private:int k;
};int main()
{CTestFClassA obj1(1, 2), obj2(3, 4);CTestFClassB objb(5);objb.disp(obj1);cout << endl;objb.disp(obj2);return 0;
}
注意事项
(1) 友元关系不能被继承。
(2)友元关系是单向的,不具有交换性。若类B是类A的友元,类A不一定是类B的友元,要看在类中是否有相应的声明。
(3)友元关系不具有传递性。若类B是类A的友元,类C是B的友元,类C不一定是类A的友元,同样要看类中是否有相应的申明。
二、动态内存
在定义变量或数组的同时即在内存为其开辟了指定的固定空间。
int n, a[10];
char str[100];
一经定义,即为固定地址的空间,在内存不能被别的变量所占用。
在程序内我们有时需要根据实际需要开辟空间,如输入学生成绩,但每个班的学生人数不同,一般将人数定得很大,这样占用内存。
示例:
#define N 1000
......
float score[N][5];
cin>>n;
for(int i=0;i<n;i++)for(j=0;j<5;j++)cin>>score[i][j];
......
无论班级中有多少个学生,程序均在内存中开辟1000×5个实型数空间存放学生成绩,造成内存空间的浪费。
利用 new 运算符可以在程序中动态开辟内存空间。
new 数据类型[单位数];
new int[4];
在内存中开辟了4个int型的数据空间,即16个字节。
new 相当于一个函数,在内存开辟完空间后,返回这个空间的首地址,这时,这个地址必须用一个指针保存下来,才不会丢失。
同样,利用new运算符也可以开辟连续的多个空间(数组)。
注意:用new开辟的内存单元没有名字,指向其首地址的指针是引用其的唯一途径,若指针变量重新赋值,则用new开辟的内存单元就在内存中“丢失”了,别的程序也不能占用这段单元,直到重新开机为止。
用 new 运算符分配的空间,不能在分配空间时进行初始化。
同样,用new开辟的内存单元如果程序不“主动”收回,那么这段空间就一直存在,直到重新开机为止。
delete运算符用来将动态分配到的内存空间归还给系统,使用格式为:delete p;
示例:
#include <iostream>
using namespace std;class CNewDelete
{
public:CNewDelete() {cout << "\n调用默认的构造函数CNewDelete::CNewDelete()." << endl;str = new char[10];}~CNewDelete() {cout << "\n调用析构函数CNewDelete::~CNewDelete()." << endl;delete[]str;}
private:char *str;
};int main()
{char buffer[100]; CNewDelete *pobj = new(buffer) CNewDelete();pobj->~CNewDelete(); // 主动调用析构函数,避免内存泄漏char *buffer2 = new char[100];CNewDelete *pobj2 = new(buffer2) CNewDelete();pobj2->~CNewDelete(); // 必须记住,主动调用析构函数delete[]buffer2; // 堆内存需要主动释放操作return 0;
}
注意,定位 new 的基本语法如下:
Type ptr = new (address) Type(constructor_arguments);*
- Type 是要创建的对象的类型。
- address 是一个指向已分配内存的指针
- constructor_arguments
- 是可选的,传递给构造函数的参数。
相关文章:
C++友元与动态内存
一、友元 友元是一种定义在类外部的普通函数或类,但它需要在类体内进行说明,为了与该类的成员函数加以区别,在说明时前面加以关键字friend。友元不是成员函数,但是它可以访问类中的私有成员。 类具有封装和信息隐藏的特性。…...
catch-all路由
介绍 ✅ 什么是 Catch-All 路由? Catch-All 路由 指的是:一个能匹配“任意路径”的通配型路由。 它一般会使用 路径参数 path 类型,比如: app.get("/{full_path:path}") async def fallback_handler(full_path: str):…...
jdk21新特性详解使用总结
jdk21新特性详解总结 1.StringBuilder和StringBuffer新增了一个repeat方法 /*** Java 21的StringBuilder和StringBuffer新增了一个repeat方法*/public static void repeatStr(){var sbnew StringBuilder().repeat("*",10);System.out.println(sb);}运行结果如下&…...
子网划分超AI教程:5分钟教会划分子网
友情提示:本文内容由银河易创AI(https://ai.eaigx.com)创作平台deepseek-v3模型生成,仅供参考 前言 子网划分(Subnetting)是网络工程师和IT运维人员必须掌握的基础技能,但对于初学者来说&#…...
制造业数字化转型:流程改造先行还是系统固化数据?基于以MTO和MTS的投资回报分析
1. 执行摘要 制造业正经历一场深刻的数字化转型,企业面临着先进行流程改造以优化运营,还是直接上线系统以固化数据的战略选择。本文深入分析了以销定产(MTO)和以产定销(MTS)两种主要生产模式下,…...
【实用技巧】电脑重装后的Office下载和设置
写在前面:本博客仅作记录学习之用,部分图片来自网络,如需引用请注明出处,同时如有侵犯您的权益,请联系删除! 文章目录 前言下载设置总结互动致谢参考目录导航 前言 在数字化办公时代,Windows和…...
使用Android 原生LocationManager获取经纬度
一、常用方案 1、使用LocationManager GPS和网络定位 缺点:个别设备,室内或者地下停车场获取不到gps定位,故需要和网络定位相结合使用 2、使用Google Play服务 这种方案需要Android手机中有安装谷歌服务,然后导入谷歌的第三方库: 例如:i…...
STM32开发板上生成PWM正弦波
在STM32开发板上生成正弦波通常需要结合定时器(TIM)、数模转换器(DAC)或脉宽调制(PWM)以及时钟系统的配置。以下是分步指南: 方法1:使用DAC 定时器(推荐) 步…...
量子计算与人工智能融合的未来趋势
最近研学过程中发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击链接跳转到网站人工智能及编程语言学习教程。读者们可以通过里面的文章详细了解一下人工智能及其编程等教程和学习方法。 在当今科技飞速发展…...
关于登录鉴权session、cookie和token
一、cookie是用来解决什么问题的? 假如现有业务需求:当浏览器发起一个url请求之后,在一个会话周期内,服务端需要判断这个用户是否第一次发起请求,第一次请求展示的页面跟第N次请求需要响应的页面不同的。现在我们大部分…...
206. 反转链表 92. 反转链表 II 25. K 个一组翻转链表
leetcode Hot 100系列 文章目录 一、翻转链表二、反转链表 II三、K 个一组翻转链表总结 一、翻转链表 建立pre为空,建立cur为head,开始循环:先保存cur的next的值,再将cur的next置为pre,将pre前进到cur的位置…...
实时内核稳定性 - scheduling while atomic
scheduling while atomic问题 根因:未成对使用获取cpu_id的函数[ 291.881071][ 0] [XW]: type=0x00000003 cpuid=4 time=1725877230 subj...
离线语音识别 ( 小语种国家都支持)可定制词组
1产品介绍 离线语音模组采用神经网络算法,支持语音识别、自学习等功能。运用此模组将 AI 技 术赋能产品,升级改造出语音操控的智能硬件 ( 例如风扇、台灯、空调、马桶、按摩椅、运 动相机、行车记录仪等 ) 。支持全球多种语言识别,如中文…...
网络华为HCIA+HCIP 策略路由,双点双向
目录 路由策略,策略路由 策略路由优势 策略路由分类 接口策略路由 双点双向 双点双向路由引入特点: 联系 路由回灌和环路问题 路由策略,策略路由 路由策略:是对路由条目进行控制,通过控制路由条目影响报文的转发路径,即路…...
【面试篇】JVM
文章目录 一、JVM 内存结构1. 请详细描述 JVM 的内存结构,各个区域的作用是什么?2. 堆内存是如何划分的?新生代和老年代的比例是多少?3. Eden 区和 Survivor 区的作用是什么?它们之间是如何协作的?4. 方法区…...
【TI MSPM0】ADC DAC学习
一、样例展示 通过ADC0触发单次采样,如果采样结果大于0.5倍的VDD,就点亮LED 否则熄灭LED 编译加载运行这个历程,提供一个电压到A0_2引脚上,电压范围在0-VCC之间同时观察LED1.在上电后,默认将ADC配置到正确的引脚模式,…...
Cesium系列:从入门到实践,打造属于你的3D地球应用
一、Cesium简介 CesiumJS 是一个开源的 JavaScript 库,它能够帮助开发者创建出具有卓越性能、高精度、出色视觉质量和易用性的世界级 3D 地球仪和地图。无论是在航空航天领域,用于模拟飞行路径和展示卫星数据;还是在智能城市中,用…...
笔记1——数据通信网络基础
一、概述 数据通信网络:由路由器、交换机、防火墙、无线设备以终端构成的网络 功能:实现数据互通 二、网络设备 交换机: 特点:距离终端用户最近的设备 作用:终端接入、二层交换机 广播域:交换机连接的终端构成一个广播…...
Linux系统程序设计:从入门到高级Day01
知识点1 【系统调用】 系统调用的概述 系统调用:内核 提供给 用户 可以 操作内核 的一组函数接口 关系:用户 借助 系统调用 操作内核 进程的空间分为:内核空间 和 用户空间 用户一般都是在用户空间操作的,但是有的时候用户需要…...
openEuler24.03 LTS下安装HBase集群
前提条件 安装好Hadoop完全分布式集群,可参考:openEuler24.03 LTS下安装Hadoop3完全分布式 安装好ZooKeeper集群,可参考:openEuler24.03 LTS下安装ZooKeeper集群 HBase集群规划 node2node3node4MasterBackup MasterRegionServ…...
关于testng.xml无法找到类的问题
问题:testng.xml添加测试类的时候飘红 解决办法: 1.试图通过自动生成testng.xml插件去解决,感觉也不是这个问题,没有尝试; 2.以为是创建包的方式不对,重新删除后新建--还是找不到 想新建类的时候发现从m…...
数据结构:探秘AVL树
本节重点 理解AVL树的概念掌握AVL树正确的插入方法利用_parent指针正确更新平衡因子掌握并理解四种旋转方式:左单旋,右单旋,左右双旋,右左双旋 一、AVL树的概念 AVL树得名于它的发明者G. M. Adelson-Velsky和E. M. Landis&…...
Linux 入门:基础开发工具(上)vim,gcc/g++,make/makefile
目录 一.软件包管理器 一).软件包 二).安装软件 三).删除软件 二.编辑器vim 一).vim的基本介绍 1.正常/普通/命令模式(Normal mode) 2.插入模式(Insert mode) 3.底行模式(last line mode) 二).vim的基本操作 …...
计算机科学基础设施之数学:科研工具、资源与环境详介
李升伟 整理 数学科研涉及广泛的工具、资源和环境,涵盖从理论分析到数值模拟、从数据获取到论文发表的各个环节。以下是对数学科研中常用工具、资源和环境的详细介绍: 一、数学科研工具 1. 文献检索与管理工具 Google Scholar:全球最大的…...
Turtle事件处理(键盘与鼠标交互)
Turtle 提供了 事件驱动编程,允许我们使用 键盘 和 鼠标 控制 Turtle,从而实现交互式绘图。例如,我们可以让 Turtle 响应 按键、鼠标点击 和 拖动 事件,使其根据用户的输入进行移动、旋转或绘制图形。 1. 事件机制概述 Turtle 的事件处理主要依赖 turtle.Screen() 提供的 …...
5、无线通信基站的FPGA实现架构
基站(Base Station,BS),也称为公用移动通信基站,是无线电台站的一种形式,具体则指在一定的无线电覆盖区中,通过移动通信交换中心,与移动电话终端之间的信息传递的无线电收发信电台。…...
关于 UPDATE 语句 和 SELECT ... FOR UPDATE 的对比分析,包括语法、功能、锁机制、使用场景及示例代码
以下是关于 UPDATE 语句 和 SELECT ... FOR UPDATE 的对比分析,包括语法、功能、锁机制、使用场景及示例代码: 1. UPDATE 语句 功能 直接修改数据:立即更新表中的数据,并提交修改。无显式锁:虽然会自动加锁ÿ…...
Linux2 CD LL hostnamectl type mkdir dudo
查看主机名信息 设置静态主机名 同时配置静态、瞬时主机名 下载Vmware tools https://blog.csdn.net/qq_34638161/article/details/102779721 mkdir创建目录 问题:为什么在root目录下 看不到 /var /usr那些文件夹...
Docker容器部署Java项目的自动化脚本(Shell编写)
🤟致敬读者 🟩感谢阅读🟦笑口常开🟪生日快乐⬛早点睡觉 📘博主相关 🟧博主信息🟨博客首页🟫专栏推荐🟥活动信息 文章目录 Docker容器部署Java项目的自动化脚本&#x…...
计组(蒋)期末不挂科纲要
2025.03.27:计算机组成原理期末不挂科速成纲要 计组期末不挂科速成纲要 第1章 概论第2章 数据的机器层次表示习题练习 第3章 指令系统习题练习 第4章 数值的机器运算习题练习 第5章 存储系统和结构习题练习 第6章 中央处理器习题练习 第7章 总线 第1章 概论 冯诺依曼…...
