C++初阶 | [三] 类和对象(中)

摘要:类的6个默认成员函数,日期类

如果一个类中什么成员都没有,简称为空类。然而,空类并不是什么成员都没有,任何类在什么都不写时,编译器会自动生成6个默认成员函数。默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数。

d830f19069b64fedaf2db4eaf7195e58.png

(ps.下文中标题右上角有数字标识的为6个默认成员函数的内容,以示区分) 


1. 构造函数_Constructor¹

构造函数:创建对象?❌ → 初始化

features

  1. 函数名同类名相同
  2. 没有返回值(void
  3. 对象实例化时编译器自动调用80ed5ff83b004172babc0c305b56a511.gif
  4. 可以实现重载
  5. 构造函数不写编译器会默认生成

注意 warning

  • 无参和全缺省会存在歧义,调用的时候不明确,建议实现全缺省构造函数。1f07bc9e33f04cf19bada572c98efdb7.png
  • 调用无参构造函数时,不要加括号,这样写同样会引起歧义,编译器分不清是对象还是函数声明 ( e.g.Date d2();  函数声明——Type function_name(parameter list(可为空))。
  • 构造函数不可以显式调用!

默认构造函数:可以不用传参的构造函数(包括自己没有显式写构造函数而编译器自动生成的,默认构造函数有且只有一个)即无参或全缺省构造函数。

自动生成的构造函数

  1. 自己不写编译器才会自动生成,写了编译器就不会再自动生成其他构造函数。
  2. 自动生成的构造函数对 内置类型 不做处理(对于自定义类型会因具体的编译器而异,语法上规定不做处理,但可能有的编译器会处理),针对这个内置类型不处理的问题,C++11支持在声明处给缺省值,其他地方都没有给初始化值就会用缺省值初始化。(注意:所有的指针都是内置类型,自定义类型的指针也是内置类型!)b6efd991cb4d47c9b168ce90f5b30f7f.png
  3. 对于 自定义类型 会自动调用其自己的默认构造函数。3e9749743fb945fb8dcbfbb74ac73635.png

sum.真正便捷的是自动调用 


2. 析构函数_Destructor²

析构函数:销毁对象?❌(出作用域生命结束编译器自动销毁) → 完成对象中的资源清理工作 ✔

features

  1. 函数名为:~类名
  2. 没有返回值(void),没有参数
  3. 有且只有一个,不可重载(都没有参数肯定无法构成函数重载)
  4. 对象生命周期结束自动调用,调用顺序遵循栈后进先出的原则,即后定义的先析构
  5. 同构造函数一样,自己不写编译器会自动生成

什么情况下需要自己写析构函数?

10367a0783aa4f84aa40190502265599.png


3. 拷贝构造_Copy Constructor³

(注:拷贝构造函数也是构造函数)

值拷贝/浅拷贝

值拷贝:将数据内容完全一样地拷贝一份。但在有些场景中,值拷贝会导致一些问题,如下图所示。

如何解决两次调用析构函数的问题?

  1. 传值传参 → 传引用传参
  2. 实现拷贝构造函数实现深拷贝,如下图(传值传参时,对于自定义类型会调用拷贝构造函数

features

  1. 拷贝构造是构造函数的一个函数重载——函数名为类名,没有返回值。
  2. 拷贝构造参数必须传引用传参(这里的引用推荐加 const 以免将要被拷贝的对象被修改)。否则会导致无穷调用调用拷贝构造函数 → 传值传参 → 值拷贝 → 调用拷贝构造 →…(示例如下图)

 正确的拷贝构造函数示例:

class Date
{
public:
	Date(const int year = 1, const int month = 1, const int day = 1)//全缺省构造函数
	{
		_year = year;
		_month = month;
		_day = day;
	}

	Date(const Date& d)//copy constructor
	{
		_year = d._year;
		_month = d._month;
		_day = d._day;
	}
private:
	int _year;
	int _month;
	int _day;
};

自动生成的默认拷贝构造

  1. 对于 内置类型 值拷贝(例如对于上述日期类不用自己写拷贝构造,值拷贝已经能够满足需求)
  2. 对于 自定义类型 自动调用拷贝构造函数

4.操作符重载_operator

class Date
{
public:
	Date(const int year = 1, const int month = 1, const int day = 1)//全缺省构造函数
	{
		_year = year;
		_month = month;
		_day = day;
	}

	bool operator<(const Date& d)//操作符重载
	{
		if (_year < d._year)
			return true;
		else if (_year == d._day && _month < d._month)
			return true;
		else if (_year == d._day && _month == d._month && _day < d._day)
			return true;
		else
			return false;
	}
private:
	int _year;
	int _month;
	int _day;
};

int main()
{
	Date d1(2022, 2, 3);
	Date d2(2022, 1, 4);

	cout << (d1 < d2)<<endl;

	return 0;
}

注:上述代码中 d1<d2 → d1.operator<(d2)  (编译器会自动转化为这样显式的调用,所以这与调用其他成员函数是一样的),隐藏的参数 this指针 指向该重载操作符的左操作数 d1,传递的参数 d2 为右操作数。

Rules

  1. 不能创造原本没有的操作符。(e.g.operator@)
  2. 参数中必须有一个自定义类型。
  3. 不能改变操作符原本的含义。
  4. 对于成员函数,第一个参数为 this 指针。即 this 指针所指向的对象一定是操作符的左操作数。
  5. 5个能重载的操作符:::     ?:    .*   sizeof    .

A major design goal of C++ is to let programmers define their own types that are as easy to use as the built-in types.

——《C++ Primer》

操作符重载使得使用自定义类型就像使用内置类型一样便捷。 


5.class Date

1)成员函数声明和定义分离:

类内声明,内外定义。

  1. 缺省值应在函数声明时给出,定义处不能再次定义缺省值,会引发重定义错误。
  2. 定义成员函数要指明类域(在返回值类型之后,函数名之前)——Type classname::function_name(parameter list)

2)constructor:

  • 注意检查日期是否合法(传参可能传递非法日期 )→ 年份没有限制,月份必须在 [1,12] 的区间内,日期受到年份(闰年)和月份的限制。

(下述代码补充说明:GetMonthDay 函数中的 arraystatic 修饰是因为考虑到会频繁调用该函数,导致频繁开辟 array 空间,所以选择将 array 的数据放置在静态区,减少重复开空间,warning:不要对该数组里的数据进行修改!数据存储在静态区,每次调用该函数,修改的行为会被不断累积!!出于这样的考虑,array 可以加 const 保护

int GetMonthDay(const int year, const int month)
{
	static int array[13] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };
	if (month == 2)
	{
		if (((year % 100 != 0) && (year % 4 == 0)) || (year % 400 == 0))
			///++array[2];//error!!!!!!!!!!!!!
            return 29;
	}
	return array[month];
}

Date::Date(const int year, const int month, const int day)
{
	if (month <= 12 && day <= GetMonthDay(month))
	{
		_year = year;
		_month = month;
		_day = day;
	}
	else
	{
		_year = _month = _day = 0;
		cout << "日期非法" << endl;
	}
}

3)赋值重载_operator=⁴

①赋值与拷贝构造的区别:

赋值重载:将一个变量中的数据 赋 给另一个变量

拷贝构造:用一个已经存在的对象来创建一个新的对象(构造函数主要是完成初始化工作,定义一个自定义类型的变量时会自动调用构造函数!

②连续赋值

赋值的结果返回Date对象(推荐传引用返回)

③ 自己给自己赋值

自己给自己赋值,即 this 指针与传引用传参的参数表示的是同一个存储空间,所以不用继续进行赋值直接返回。综合上述所说,示例代码如下:

const Date& Date::operator=(const Date& d)
{
	if (this == &d)
		return *this;

	_year = d._year;
	_month = d._month;
	_day = d._day;
	return d;
}

④自动生成默认赋值重载函数 

同构造函数和析构函数一样,自己不写会自动生成。自动生成的默认赋值重载函数会完成值拷贝对于日期类,值拷贝已经可以满足需求,因此可以不用自己写赋值重载函数。

4)其他常用操作符重载:

①+= 与 +

  1. 复用的问题:①operator+ 复用 operator+= 两次调用拷贝构造;②operator+= 复用 operator+ 四次调用拷贝构造,一次赋值。所以选择operator+ 复用 operator+= 更好。
  2.  += 负数:注意传参可能为负值的情况
  3.  处理月份的注意:月份是从 [1,12] 的循环

示例代码:

Date& Date::operator+=(int days)
{
	if (day < 0)
		return ((*this) -= -day);

	_day += days;
	int _month_max_day = GetMonthDay(_year, _month);
	while (_day > _month_max_day)
	{
		_day -= _month_max_day;

		//处理月份
		++_month;
		if (_month > 12)
		{
			++_year;
			_month = 1;
		}

		//更新月最大天数
		_month_max_day = GetMonthDay(_year, _month);

	}
	return *this;
}

Date Date::operator+(const int days) const
{
	Date tmp(*this);
	tmp += days;
	return tmp;
}

②前置++ 与 后置++

后置++ 的操作符重载中,语法规定通过参数 int 与前置++构成函数重载,int 只是用来占位,具体传什么内容由编译器处理。例如如下代码的函数声明:(再次提醒只是返回值不同不能构成函数重载)

Date& operator++();//前置++
Date operator++(int);//后置++

5)日期 - 日期 运算

  • 方式一:直接算
  • 方式二:让 小日期++ 直到 等于 大日期,++多少次就相差多少天(其中要用到比较大小的操作符,关于这些操作符的重载在本文第4部分——操作符重载——有一处示例,实现其中一个操作符,其他可以复用,思路简单,在此不多做赘述。以下提供一个实现日期相减的函数示例)
    int Date::operator-(const Date& d) const
    {
    	int count = 0, flag = 1;
    	Date Max_date = *this;
    	Date Min_date = d;
    	if (d > *this)
    	{
    		flag = -1;
    		Max_date = d;
    		Min_date = *this;
    	}
    	while (Min_date < Max_date)
    	{
    		++count;
    		++Min_date;
    	}
    	return count*flag;
    }
    

6. const 对象

在 C++ 入门一文中在介绍常引用(const 引用)时,讨论了关于权限放大的问题,同样,对于 const对象 来说,它们无法调用 非const成员函数——因为会导致权限放大的问题。

 由上图不难看出,一个函数 const 版本非const版本 可以构成函数重载——参数类型不同,this指针的类型分别是 const Type* const Type* const (例如,对于operator[]的操作符重载就很有必要实现 const 和 非const 两个版本)

  • 建议:不做修改的成员函数都加 const 修饰,注意互相复用的函数之间会相互影响!

7. 取地址重载⁵ 及 const 对象取地址⁶

这两个默认成员函数一般不用重新定义 ,编译器默认会生成。这两个运算符一般不需要重载,使用编译器生成的默认取地址的重载即可,只有特殊情况,才需要重载,比如想让别人获取到指定的内容

class Date
{
public:
	const Date* operator&()const
	{
		return nullptr;
	}

private:
	int _year;
	int _month;
	int _day;
};

 8. 补充:流插入和流提取操作符重载

 以上,我们可以了解到,流提取 >> 的操作数为 cin —— 一个类型为 istream 的对象,右操作数为一个内置类型的变量,同理,流插入 << 的操作数为 cout —— 一个类型为 ostream 的对象,右操作数为一个内置类型的变量。

接下来,我们尝试对日期类的对象实现流插入和流提取的操作符重载(以流提取 >> 为例,流插入同理):因为日期类的成员变量私有在类外无法访问,所以我们尝试在类内实现。同时,考虑到连续流插入/流提取,返回值类型应为ostream/istream。如下代码。

class Date
{
public:  
    ostream& operator<<(ostream& out)
    {
	    out << _year << "/" << _month << "/" << _day << endl;
	    return out;
    }
private:
	int _year;
	int _month;
	int _day;
};

在上述代码中, 可以看到,第一个参数为 隐藏的this指针 ,第二个参数为 ostream对象的引用。则调用该函数即为:d(Type:Date)<<cout(Type:ostream),显然,这样的实现是符合该操作符原本的使用习惯的,在类内实现第一个参数必为 隐藏的this指针,所以考虑在类外实现该函数,为了能够访问内部成员变量,需要用到友元声明(详见类和对象下)。

class Date
{
public:
	friend ostream& operator<<(ostream& out, Date d);

private:
	int _year;
	int _month;
	int _day;
};


ostream& operator<<(ostream& out, Date d)
{
	out << d._year << "/" << d._month << "/" << d._day << endl;
	return out;
}

cin/cout 的意义:针对自定义类型可以实现重载(scanf/printf 不能很好地支持自定义类型);能够自动识别类型(例如对于 printf 函数如果数据类型改变,相应地也要改变打印格式)


END

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

python中列表的基础解释

列表&#xff1a; 一种可以存放多种类型数据的数据结构 列表的创建&#xff1a; 1.用【】创建列表 #创建一个空列表 list1[] #创建一个非空列表 list2 [zhang,li,ying,1,2,3] #输出内容及类型 print(list1,type(list1)) print(list2,type(list2))结果&#xff1a; 2.使用list…

《视觉SLAM十四讲》-- 回环检测

文章目录 10 回环检测10.1 概述10.1.1 回环检测的意义10.1.2 回环检测的方法10.1.3 准确率和召回率 10.2 词袋模型10.3 字典10.3.1 字典的结构10.3.2 实践&#xff1a;创建字典 10.4 相似度计算10.4.1 理论部分10.4.2 实践&#xff1a;相似度的计算 10.5 实验分析与评述 10 回环…

股票价格预测 | Python实现基于CNN卷积神经网络的股票预测模型(keras,Conv1D)

文章目录 效果一览文章概述源码设计参考资料效果一览 文章概述 股票价格预测 | Python实现基于CNN卷积神经网络的股票预测模型(keras) 源码设计 import quandl import datetimedf = quandl

FreeRTOS(教程非常详细)

概述&#xff1a; 之前写了关于FreeRTOS的部分内容&#xff0c;为了方便阅读&#xff0c;现在给汇总到一起了。全部学习完后&#xff0c;恭喜你对FreeRTOS有了更深的认知。 第一章 FreeRTOS移植到STM32 第二章 FreeRTOS创建任务 第三章 FreeRTOS任务管理 第四章 FreeRTOS消…

LeetCode【12】整数转罗马数字

题目&#xff1a; 思路&#xff1a; https://blog.csdn.net/m0_71120708/article/details/128769894 代码&#xff1a; public String intToRoman(int num) {String[] thousands new String[] {"", "M", "MM", "MMM"};String[] hun…

【自然语言处理】【大模型】赋予大模型使用工具的能力:Toolformer与ART

赋予大模型使用工具的能力&#xff1a;Toolformer与ART ​ 本文介绍两种赋予大模型使用外部工具能力的方法&#xff1a;Toolformer和ART。 Toolformer论文地址&#xff1a;https://arxiv.org/pdf/2302.04761.pdf ART论文地址&#xff1a;https://arxiv.org/pdf/2303.09014.pd…

【网络奇遇记】那年我与计算机网络的浅相知

&#x1f308;个人主页&#xff1a;聆风吟 &#x1f525;系列专栏&#xff1a;网络奇遇记、数据结构 &#x1f516;少年有梦不应止于心动&#xff0c;更要付诸行动。 文章目录 一. 计算机网络的定义1.1 计算机早期的一个最简单的定义1.2 现阶段计算机网络的一个较好的定义 二. …

【IPC】消息队列

1、IPC对象 除了最原始的进程间通信方式信号、无名管道和有名管道外&#xff0c;还有三种进程间通信方式&#xff0c;这 三种方式称之为IPC对象 IPC对象分类&#xff1a;消息队列、共享内存、信号量(信号灯集) IPC对象也是在内核空间开辟区域&#xff0c;每一种IPC对象创建好…

【汇编】处理字符问题

文章目录 前言一、处理字符问题1.1 汇编语言如何处理字符1.2 asciiascii码是什么&#xff1f;ascii码表是什么&#xff1f; 1.3 汇编语言字符示例代码 二、大小写转换2.1 问题&#xff1a;对datasg中的字符串2.2 逻辑与和逻辑或2.3 程序&#xff1a;解决大小写转换的问题一个新…

「项目阅读系列」go-gin-example star 6.5k!(1)

文章目录 准备工作适宜人群项目信息 项目结构代码阅读主要模块代码主函数模块router 路由模块auth 授权模块数据库 修改文章请求分析其他依赖 总结 准备工作 适宜人群 初学 go 语法&#xff0c;希望了解 go 项目的构建过程和方式。 项目信息 go-gin-example 项目是使用 gin…

qt-C++笔记之两个窗口ui的交互

qt-C笔记之两个窗口ui的交互 code review! 文章目录 qt-C笔记之两个窗口ui的交互0.运行1.文件结构2.先创建widget项目&#xff0c;搞一个窗口ui出来3.项目添加第二个widget窗口出来4.补充代码4.1.qt_widget_interaction.pro4.2.main.cpp4.3.widget.h4.4.widget.cpp4.5.second…

JAVA for 循环训练 Pattern

import java.util.Scanner;public class Pattern {public static void main(String[] args) {int[] arr {0, 1, 1, 0, 1, 0, 0, 1, 1, 0, 0, 1, 0, 1, 1, 0};Scanner in new Scanner(System.in);System.out.print("请输入n:");int n in.nextInt();in.close();for …

LeetCode27.移除元素(暴力法、快慢指针法)

每日一题&#xff1a;LeetCode27.移除元素 1.问题描述2.解题思路3.代码 1.问题描述 问题描述&#xff1a;给你一个数组 nums 和一个值 val&#xff0c;你需要 原地 移除所有数值等于 val 的元素&#xff0c;并返回移除后数组的新长度。不要使用额外的数组空间&#xff0c;你必…

Linux(Ubuntu)安装JDK环境

系统环境 Ubuntu20.04 下载JDK压缩包 前往Oracle官网进行后续下载或单击下载JDK压缩包 下拉找到JDK8&#xff0c;在Linux板块下选择适配系统架构的压缩包文件(后缀为tar.gz)&#xff0c;系统架构可通过uname -m命令查看 安装JDK 安装环境通常放在/usr/local下&#xff0c;进入…

免费稳定几乎无门槛,我的ChartGPT助手免费分享给你

公众号「架构成长指南」&#xff0c;专注于生产实践、云原生、分布式系统、大数据技术分享。 概述 ChatGPT想必大家应该都不陌生了&#xff0c;大部分人或多或少都接触了&#xff0c;好多应该都是通过openAi的官方进行使用的&#xff0c;这个门槛对大部分人有点高&#xff0c;…

公共字段自动填充-@TableField的fill实现(2)

TheadLocal 客户端发送的每次http请求&#xff0c;在服务端都会分配新的线程。因此登录检查过滤器、controller、元数据对象处理器属于一个线程。 TheadLocal是线程的局部变量&#xff1a; TheadLocal常用方法&#xff1a; 如何在元数据对象处理器中获取当前登录用户的id&…

开发知识点-uniapp微信小程序-开发指南

uniapp uni.chooseLocationgetCurrentPages美团外卖微信小程序开发uniapp-美团外卖微信小程序开发P1 成果展示P2外卖小程序后端&#xff0c;学习给小程序写http接口P3 主界面配置P4 首页组件拆分P13 外卖列表布局筛选组件商家 布局测试数据创建样式 请求商家外卖数据封装请求并…

酷柚易汛ERP - 序列号盘点操作指南

1、应用场景 将系统中开启序列号的商品数量与与实际存放的数量进行对比。 2、主要操作 2.1 录入序列号 打开【盘点】-【序列号盘点】&#xff0c;新增序列号盘点单&#xff0c;点击【SN】按钮&#xff0c;在弹框中输入序列号。 支持扫描枪录入序列号支持复制粘贴序列号录入…

JS特效:跟随鼠标移动的小飞机

前端网页中&#xff0c;用JS实现鼠标移动时&#xff0c;页面中的小飞机向着鼠标移动。 效果 源码 <!DOCTYPE html> <html><head><style>*{margin: 0;padding: 0;}body{height: 100vh;background: linear-gradient(200deg,#005bea,#00c6fb);}#plane{…

【WiFI问题自助】解决WiFi能连上但是没有网的问题

WiFi能连上但是没有网的问题 背景&#xff1a;wifi能连上&#xff0c;但是没有网 解决 遇事不决&#xff0c;先重启啊&#xff01;怎么重启&#xff1f;拔掉电源再插上&#xff01;拔掉网线再插上&#xff01; 直接ok了。 思考记录 今天WiFi又上不了网了&#xff0c;昨天报…