从0开始C++(五):友元函数运算符重载

友元函数

介绍

C++中的友元函数是一种特殊的函数,它可以访问和操作类的私有成员和保护成员。友元函数可以在类的内部或外部声明和定义,但在其声明和定义中需要使用关键字 friend 来标识。友元函数可以是全局函数,也可以是其他类的成员函数。

下面是友元函数的一些重要特点和用法:

  1. 友元函数可以访问和操作类的私有成员和保护成员,这使得函数可以直接访问类的私有数据,而无需通过类的公有接口进行访问。

  2. 友元函数可以在类的内部或外部声明和定义。在类的内部声明友元函数时,使用friend关键字来标识函数为友元函数。在类的外部定义友元函数时,需要在函数的定义前面声明该函数为友元函数。

  3. 友元函数可以是全局函数,也可以是其他类的成员函数。如果友元函数是另一个类的成员函数,需要在该类的声明中将其声明为友元函数。

  4. 友元函数不是类的成员函数,因此它没有隐含的this指针,也无法访问类的非静态成员。

  5. 友元函数的声明可以放在类的任何位置,但其定义必须在类的外部。这是因为友元函数不属于类的成员,所以需要在类的外部定义。

  6. 友元函数可以用于重载操作符。这使得我们能够定义两个不同类对象之间的操作,例如添加两个自定义类的对象。

友元函数提供了一种特殊的访问和操作类的私有成员和保护成员的方式。它在某些情况下可以简化代码实现,但同时也破坏了封装性,因此需要谨慎使用。

创建友元函数

要创建一个友元函数,需要按照以下步骤进行操作:

1、在类内声明友元函数:在类的声明中,使用 friend 关键字来声明友元函数。友元函数可以是全局函数,也可以是其他类的成员函数。例如,下面的代码将一个全局函数声明为类的友元函数:

class MyClass {
public:// 声明友元函数friend void myFriendFunction();
};

2、在类的外部定义友元函数:在类的外部,需要定义先前声明的友元函数。这样,友元函数就可以访问和操作类的私有成员和保护成员。例如,下面的代码定义了先前声明的友元函数:

// 定义友元函数
void myFriendFunction() {// 在此函数中可以访问和操作MyClass的私有成员和保护成员
}

3、使用友元函数:使用类的对象或类名和作用域解析运算符(::)调用友元函数。例如,下面的代码演示了如何使用类对象调用友元函数:

MyClass obj;
myFriendFunction();  // 使用类对象调用友元函数

或者,如果友元函数是另一个类的成员函数,可以使用该类的对象或类名和作用域解析运算符(::)调用友元函数。例如,下面的代码演示了如何使用类对象调用另一个类的成员函数作为友元函数:

class MyClass2 {
public:// 声明友元函数friend void MyClass::myFriendFunction2();
};// 定义友元函数
void MyClass::myFriendFunction2() {// 在此函数中可以访问和操作MyClass2的私有成员和保护成员
}MyClass obj;
MyClass2 obj2;
obj.myFriendFunction2();  // 使用类对象调用友元函数

 运算符重载

运算符重载是指在类中重定义运算符的行为,使其适用于特定的类对象。运算符重载可以使得类对象的使用更加自然和方便,同时也提供了更多的灵活性。

要重载运算符,需要定义一个类成员函数或全局函数来执行所需操作,该函数的名称是运算符本身,并使用 operator 关键字声明。例如,要重载"+"运算符,可以定义一个名为operator+()的函数。

以下是一个示例,演示了如何重载"+"运算符:

class MyClass {
private:int value;
public:MyClass(int v) : value(v) {}MyClass operator+(const MyClass& other) {  // 运算符重载函数return MyClass(value + other.value);}int getValue() const {return value;}
};int main() {MyClass obj1(3);MyClass obj2(5);MyClass result = obj1 + obj2;  // 使用重载的"+"运算符std::cout << result.getValue() << std::endl;  // 输出结果:8return 0;
}

上述示例中,我们在MyClass类中重载了"+"运算符。重载函数声明中的参数是另一个MyClass对象(const引用),并返回一个新的MyClass对象。在main函数中,我们创建了两个MyClass对象,然后使用重载的"+"运算符将它们相加,最终得到一个新的MyClass对象。

需要注意的是,在重载运算符时,可以根据需要选择将函数定义为类成员函数全局函数(定义全局函数的时候需要将其声明为友元函数)。对于一些需要访问类私有成员的运算符,比如赋值运算符"=",通常选择将函数定义为类成员函数;对于一些不需要直接访问类私有成员的运算符,可以选择将函数定义为全局函数。

运算符重载可以大大简化代码,并提高可读性和可维护性。但是,需要谨慎使用运算符重载,以避免混淆和误用。同样,不应该过度使用运算符重载,以保持代码的简洁和易读性。

特殊运算符重载

1、赋值运算符重载
除了之前学习无参构造函数拷贝构造函数析构函数之外如果程序员手写编译器自动添加一个赋值运算符重载, 赋值运算符只能使用成员函数实现重载
当类中出现指针类型成员变量默认赋值运算符重载函数类似于默认浅拷贝构造函数因此也需要手动编写解决浅拷贝问题

【常见面试题】一个类如果什么都不编译添加了那些代码

无参构造函数拷贝构造函数析构函数赋值运算符重载函数

2、类型转换运算符重载

类型转换运算符重载允许将一个类的对象转换为另一个类型。

类型转换运算符重载的语法如下:

operator type() {// 进行类型转换的操作
}

其中,type是要转换的目标类型。重载的类型转换运算符可以是类的成员函数或类的非成员函数。

下面是一个示例,演示了如何重载类型转换运算符:

class MyClass {
private:int value;
public:MyClass(int v) : value(v) {}operator int() {return value;}
};int main() {MyClass obj(42);int result = obj;  // 使用重载的类型转换运算符将MyClass对象转换为intstd::cout << result << std::endl;  // 输出结果:42return 0;
}

在上述示例中,我们定义了一个名为MyClass的类,并在其中重载了类型转换运算符。重载函数声明中的返回类型是要转换的目标类型(int),并在函数体中返回了MyClass对象的value成员。

在main函数中,我们创建了一个MyClass对象,并将其赋值给一个int类型的变量。由于我们已经重载了类型转换运算符,所以可以直接将MyClass对象转换为int类型。最终,输出结果为42。

需要注意的是,在重载类型转换运算符时,应该在函数体中进行适当的类型转换操作,并返回目标类型的对象。此外,需要慎谨使用类型转换运算符,以避免混淆和误用。

另外,C++还提供了显式转换运算符重载(explicit conversion operator),通过在重载函数前加上 explicit 关键字来实现。这样,在使用显式转换运算符进行类型转换时,需要明确使用转换运算符进行转换。

注意事项

● 重载运算符限制在C++语言中已有运算符范围不能创建新的运算符

● 运算符重载本质上也是函数重载但是不支持函数参数默认值

● 之后运算符不能改变运算符优先级结合不能改变运算符操作数语法结构

● 运算符重载必须基于包含自定义类型不能改变基本数据类型运算规则

● 重载功能应该与原有功能类似避免没有目的滥用运算符重载

● 一般情况下双目运算符建议使用友元函数运算符重载单目运算符建议使用成员函数运算符重载

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

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

相关文章

使用Sentinel进行服务调用的熔断和限流管理(SpringCloud2023实战)

你好&#xff0c;这里是codetrend专栏“SpringCloud2023实战”。 本文简单介绍SpringCloud2023中使用Sentinel进行限流管理。 前言 随着微服务的流行&#xff0c;服务和服务之间的稳定性变得越来越重要。 Sentinel 是面向分布式、多语言异构化服务架构的流量治理组件&#…

【AI原理解析】— 盘古大模型

目录 一、模型概述 二、技术原理 1. 深度学习框架 2. 数据与训练 3. 模型架构 4. 并行训练与优化 三、创新点 四、应用场景 五、挑战与机遇 一、模型概述 定义&#xff1a;盘古大模型是由华为公司开发的一款基于深度学习和自然语言处理技术的中文AI模型。目标&#x…

Web APIs--Dom获取属性操作

目录 1.DOM&#xff08;操作网页内容、用户交互&#xff09; 2.DOM对象获取&#xff08;querySelect(‘’)、querySelectAll(‘’)&#xff09; 总结&#xff1a; 3.操作元素内容&#xff08;修改元素的文本更换内容&#xff09; 1. 元素innerText 属性 2.元素.innerHTML…

VLOOKUP和INDEX(MATCH)区别

VLOOKUP和INDEX(MATCH)都是Excel中用于查找和检索数据的函数&#xff0c;但它们有一些关键的区别&#xff1a; VLOOKUP&#xff08;垂直查找&#xff09;函数在表格的列中搜索特定的值&#xff0c;并返回相同行的另一列中的值。它只能从左到右搜索&#xff0c;即只能在查找列的…

第一百一十六节 Java 面向对象设计 - Java 终止块

Java 面向对象设计 - Java 终止块 ​try ​块也可以有零个或一个​ finally​ 块。 ​finally ​块总是与 ​try ​块一起使用。 语法 使用 ​finally​ 块的语法是 finally {// Code for finally block }​finally​ 块以关键字 ​finally​ 开始&#xff0c;后面紧跟一对…

深入分析 Android BroadcastReceiver (四)

文章目录 深入分析 Android BroadcastReceiver (四)1. 广播接收器的深入优化与应用1.1 实时性要求高的应用1.1.1 示例&#xff1a;音乐播放器中处理耳机插拔事件1.1.2 动态注册接收器 1.2 处理耗时操作1.2.1 示例&#xff1a;使用 IntentService 处理耗时操作 1.3 安全性管理1.…

探索WebKit的奥秘:打造高效、兼容的现代网页应用

WebKit是一款开源的网页浏览器引擎,支撑了包括Safari在内的多个浏览器。其高效和兼容性使其成为现代网页开发的一个重要工具。本文将深入探讨WebKit的工作原理、使用方法和优化技巧,帮助你打造高效、兼容的现代网页应用。 一、什么是WebKit? 1.1 WebKit的起源 WebKit最初…

【机器学习】深度学习赋能:基于 LSTM 的智能日志异常检测

目录 1. LSTM 简介 2. 日志序列异常检测概述 3. 数据预处理 3.1 日志解析 3.2 数据清洗 3.3 序列化 3.4 特征提取 示例代码 4. 构建 LSTM 模型 4.1 模型结构 4.2 模型构建示例 5. 训练 LSTM 模型 5.1 数据准备 5.2 模型训练 示例代码 6. 异常检测 6.1 异常分数…

处理文本内容的命令和正则表达式

处理文本内容的命令 正则表达式匹配的是文本内容&#xff0c;linux的文本三剑客 都是针对文本内容 文本三剑客&#xff1a; grep 过滤文本内容 sed 针对文本内容进行增删改查 awk 按行取列 文本三剑客都是按行进行匹配。 grep grep的作用就是使用正则表达式来匹配文本内…

虚拟现实环境下的远程教育和智能评估系统(十一)

视频帧画面知识点区域划分 知识点区域精确分割技术: 在深度学习检测模型结果基础上使用基于交并比&#xff08;IoU&#xff09;阈值的目标合并算法&#xff0c;合并过度重合目标区域面积&#xff0c;实现知识点区域精确分割 多模态知识点内容匹配策略: 图像&#xff1a;利用…

Zookeeper-02

客户端回调Zookeeper Watcher 客户端回调Zookeeper Watcher是指在Zookeeper客户端与Zookeeper服务器之间的通信过程中&#xff0c;当某种特定事件发生时&#xff0c;Zookeeper服务器会通知客户端&#xff0c;客户端在收到通知后会调用事先注册的Watcher对象的回调方法。 Watc…

【第18章】Vue实战篇之登录界面

文章目录 前言一、数据绑定1. 数据绑定2. 数据清空 二、表单校验1. 代码2. 展示 三、登录1.登录按钮2.user.js3. login 四、展示总结 前言 上一章完成用户注册&#xff0c;这一章主要做用户登录。 一、数据绑定 登录和注册使用相同的数据绑定 1. 数据绑定 <!-- 登录表单 -…

紧凑型计算微型仿生复眼

欢迎关注&#xff1a;GZH《光场视觉》 图1 研制的计算微型复眼的成像原理 1. 导读 微型曲面复眼由于具有大视场成像、大景深成像、体积较小的优势&#xff0c;在机器视觉、无人机导航、生物灵感机器人等领域引起了广泛关注。然而&#xff0c;传统的微型曲面复眼存在设计/加工…

HTTP详细总结

概念 HyperText Transfer Protocol&#xff0c;超文本传输协议&#xff0c;规定了浏览器和服务器之间数据传输的规则。 特点 基于TCP协议: 面向连接&#xff0c;安全 TCP是一种面向连接的(建立连接之前是需要经过三次握手)、可靠的、基于字节流的传输层通信协议&#xff0c;在…

SpringMvc—域对象共享数据和视图

一、向request域创建对象 先创建首页&#xff1a; 在testController这个类中&#xff1a; package com.pon.controller; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; Controller public class test…

response.setHeader用法总结

response.setHeader用法总结 大家好&#xff0c;我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01; 在Web开发中&#xff0c;response.setHeader是一个用于设置HTTP响应头的方法。通过…

MySQL的LIFO如何实现

MySQL本身并没有直接提供LIFO&#xff08;后进先出&#xff09;的数据结构或操作&#xff0c;但你可以通过某些表设计和查询来实现LIFO的效果。以下是一个基于时间戳或自动递增ID的LIFO实现方法&#xff1a; 1. 表设计 假设你有一个表my_queue&#xff0c;它有一个自增ID和一…

后台管理系统登录业务分析(图片验证码登录)

文章目录 1、登录业务分析2、登录开发流程2.1、获取图片验证码接口业务2.2、CodeImgServiceImpl2.2.1、响应 2.3、IndexController2.4、SysUserServiceImpl2.5、SysUserMapper.xml 3、springmvc拦截器创建&注册3.1、springmvc拦截器的创建3.2、springmvc拦截器注册3.3、Sys…

【Oracle APEX开发小技巧1】转换类型实现显示小数点前的 0 以 及常见类型转换

在 apex 交互式式网格中&#xff0c;有一数值类型为 NUMBER&#xff0c;保留小数点后两位的项&#xff0c;在 展示时小数点前的 0 不显示。 效果如下&#xff1a; 转换前&#xff1a; m.WEIGHT_COEFFICIENT 解决方案&#xff1a; 将 NUMBER&#xff08;20&#xff0c;2&#xf…

Vue 自定义ElementUI的Loading效果

import { loadingText, messageDuration } from "/settings";import { Loading } from "element-ui"; // loadingText、messageDuration 这两个参数我是调的公共配置文件,按自己需求来 const install (Vue, opts {}) > {/* 全局多彩Loading加载层 *…