【C++】-对于自定义类型的输入输出运算符重载

💖作者:小树苗渴望变成参天大树
❤️‍🩹作者宣言:认真写好每一篇博客
💨作者gitee:gitee
💞作者专栏:C语言,数据结构初阶,Linux,C++


前言

今天根据上篇讲的日期类的实现,来简单的讲解一下大概怎么重载输入输出运算符,其他类也可以参考这个方法来使用这个,那我们开始进入正文的讲解


一、案例引入

接上一篇,我们写了好多运算符重载,我们将结果用一个成员函数给打印出来了
在这里插入图片描述
对于C++我们的输入输出使用的都是cin和cout,因为他能实现对内置类型的输入输出很方便,对于自定义类型我们需要使用成员函数,对象再来调用就显得格格不入,想要实现像内置类型的那种我们就需要进行运算符重载,这样就可以直接使用cin和cout.

二、<<的重载

我们真正需要重载的是流插入和流提取运算符,cout只是他其中的一个操作数,我们目前想要实现的效果是这样的:

cout<<d1;

类似于:

d1+d2;

这样的二元运算符只有两个操作数

我们再来写一个类:

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

我们的cout和cin分别再类osream和istream类里面,所以我们每次都需要导入
< iostream>和std的展开
在这里插入图片描述

我们再按照之前运算符的写法再类里试着重载一下流插入:

void operator<<(ostream& cout)
{
	cout << _year <<"年" << _month <<"月" << _day <<"日" << endl;//这是内置类型所以可以直接这么写
}

在这里插入图片描述
这样居然不行,原因是运算符的左操作数默认传给隐藏的this,我们按照下面这么些就可以了

d1<<cout;

但是这么写太别扭了,我们只能将这个函数写成全局的了,传两个参数就好了:

void operator<<(ostream& cout, const date& d)
{
	cout << d._year << "年" << d._month << "月" << d._day << "日" << endl;
}

在这里插入图片描述
这样写访问不了成员变量,有两种解决办法
==1.==使用函数将成员变量返回出来
在这里插入图片描述

在这里插入图片描述

==2.==使用友元函数
在这里插入图片描述
在这里插入图片描述

大家有没有发现我们使用第一种和第二种的时候,第二个参数有的加了const,有的没加,原因就是要理解隐含指针的类型,以此类为例,隐含的this类型是date*,而这个是对象的地址,对象加const那么取地址之后的类型应该为const date*,当你第一种方法加const,那d.getyear(),就相当于d.getyear(const date*d),而getyear()成员函数的参数类型应该是date*的,所以出现参数不匹配的情况会报错,像要解决这个问题,要不去点const,要不再成员函数前面加const修改就行了,调用内置类型就不会出现这样的问题。

下篇写类和对象中篇完结的时候我在重点讲解一下,需要再理解我写的C++入门这篇博客关于引用的知识点,const修饰的变量,引用怎么使用,把那个弄懂,应该问题不大

但是这个写的还不是特别完美,我们想要达到像输出内置类型那样,一次输出多个数据

cout<<d1<<d2<<d3;

在这里插入图片描述
我们想要达到上面这种效果就要知道为什么可以这么写,再之前连续赋值的运算符那一块,我讲过是从有往左赋值,有返回值,那么对于这个是从左往右返回值应该为cout的类型,不然没办法往右运行

ostream& operator<<(ostream& cout, const date& d)
{
	cout << d._year << "年" << d._month << "月" << d._day << "日" << endl;
	return cout;
}

我这里是通过友元来访问成员变量的,因为cout也是对象,出来函数体没有被销毁,所以可以使用引用返回。

说明:对于第一个参数,我为什么没有加const修饰,原因是这是流插入运算符,往cout里面插入数据,所以不能使用从const修饰。

三、>>的重载

我们刚才讲解了流插入运算符,现在按照上面的函数,来写流提取操作符,这个也要写成全局函数,使用友元函数,上面一样的道理,也可以连续输入好几个数据,也有返回值,cin是类istream的对象,我们来看看具体函数形式

istream& operator>>(istream& cin, date& d)
{
	cin >> d._year >> d._month >> d._day;
	return cin;
}

第二参数不加入const大家应该明白,第一个参数,刚才说对于流插入,是不用加,但是这里是流提取为什么也不需要加const呢??原因是,cin里面有的数据也还是会发生改变的,不是完全不变的,所以不加const。

对于日期类,我们再上篇也讲过有的日期输入不合理,再上面我们会修改构造函数里面的内容,加一个判断就好了,这里也是一样的,但有一个细节要注意:

istream& operator>>(istream& cin, date& d)
{
	int year, month, day;
	cin >> year >> month >> day;
	if (day > 0 && day <= date::getmonthday(year, month)&&month>0&&month<13)
	{
		d._year = year;
		d._month = month;
		d._day = day;
	}
	else
	{
		cout << "非法输入" << endl;
	}
	return cin;
}

注意的点就是,我们再之前写过一个获取月份天数的函数,但是睡哦可以写成全局也可以写在类里面,对于写在全局的可以直接调用,但是写在类里面必须使用对象名去调用,也可以再函数前面加一个static来使用类名来调用,加static那么就没有隐藏的this了。这个再后面再重点介绍。

四、总结

对于这个重载好多类都是可以参照这个来写,里面的内容可能要调整,根据具体的类来写,这里面的友元函数我在类和对象下篇会再具体介绍的,这里也相当于铺垫了,大家现在不懂没有关系,知道就行了,我们今天这篇博客不难,就是细节比较多,大家要注意好,我们下篇再见
在这里插入图片描述

  • 11
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

橘柚!

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值