【C++】拷贝构造函数与运算符重载

写在前面

拷贝构造函数、赋值运算符重载、取地址运算符都是属于类的默认成员函数!
默认成员函数是程序猿不显示声明定义,编译器会中生成。

在程序编写中,我们也经常使用拷贝的方式来获取到对应的值,例如整形变量拷贝int a = 0; int b = a;等等。在程序的编写中,我们也会需要进行对象的拷贝,这样来获取到对应对象的值。


文章目录

  • 写在前面
  • 一、拷贝构造函数
    • 1.1、若未显式定义,编译器会生成默认的拷贝构造函数。
  • 二、运算符重载
    • 2.1、赋值运算符重载(类的默认成员函数)
    • 2.2、重载运算符中的特殊:前置++和后置++的重载
    • 2.3、流插入与流提取的重载运算符
    • 2.4、重载取地址操作符(是类的默认成员)


一、拷贝构造函数

拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用。是一个构造函数

拷贝构造函数也是特殊的成员函数,其特征如下:

  1. 拷贝构造函数是构造函数的一个重载形式
  2. 拷贝构造函数的参数只有一个且必须是类类型对象的引用,使用传值方式编译器直接报错,因为会引发无穷递归调用。

在这里插入图片描述
我们先创建一个标准的拷贝构造函数

class Date
{
public:
	Date() {
		cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}

	Date(const Date& d1) { //拷贝构造函数
		_year = d1._year;
		_month = d1._month;
		_day = d1._day;
	}

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

int main() {
	Date d1(2024,1,1);
	printf("d1:>");
	d1.Print();
	
	Date d2(d1);
	printf("d2:>");
	d2.Print();

	return 0;
}

程序运行结果:
在这里插入图片描述

如果我们不使用类引用作为参数,而是使用传值的方式作为参数的话,会导致无穷递归调用
在这里插入图片描述

class Date
{
public:
	Date() {
		cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}

	Date(Date d1) { //拷贝构造函数,使用传值方式
		_year = d1._year;
		_month = d1._month;
		_day = d1._day;
	}

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

int main() {
	Date d1(2024,1,1);
	printf("d1:>");
	d1.Print();
	
	Date d2(d1);
	printf("d2:>");
	d2.Print();

	return 0;
}
  • 我们知道,函数参数使用传值,那么形参就是实参的一份临时拷贝。

C++规定的拷贝:

  1. 内置类型直接拷贝
  2. 自定义类型必须调用该自定义类型对应的拷贝构造函数完成拷贝。
  • 此时,形参stack类的对象,我们实参也是stack类的对象d1,形参接收就需要进行拷贝,而这个时候又涉及到了类的拷贝,那么也是调用stack类的拷贝构造函数,但是拷贝构造函数又时需要传值……无限套娃,如下图在这里插入图片描述

程序运行结果:
在这里插入图片描述

  • 编译器会检查拷贝构造函数是否正确的编写。

在拷贝构造中,参数要加上const,加上const的好处:

  1. 防止在拷贝构造中写反,导致原对象被修改为其他值
  2. 如果原对象是const修饰的对象,也可以进行拷贝构造,不会造成权限放大。
  3. 可以作为传值返回函数的接收对象。在这里插入图片描述
class Date
{
public:
	Date() {
		//cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}
	Date(const Date& d1) {
		cout << "Date(const Date& d1)" << endl;
		_year = d1._year;
		_month = d1._month;
		_day = d1._day;
	}

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

Date getDate() {
	Date d(2222, 1, 1);
	return d;	
}

int main() {
	Date d1(2022, 5, 4);
	
	Date d2 = getDate();
	return 0;
}
  • 在拷贝构造的形参中加入const后,就可以使用getDate();函数的返回值,作为拷贝构造的形参。
  • 我们知道函数的返回值返回的是一个临时变量临时变量中具有常性。所以我们在拷贝构造函数中加入const 后就可以接收常性的参数

在这里插入图片描述

1.1、若未显式定义,编译器会生成默认的拷贝构造函数。

默认的拷贝构造函数对象按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝。

拷贝构造对类型的处理:

  1. 对内置类型成员进行值拷贝/浅拷贝
  2. 对自定义类型成员会调用它的拷贝构造函数

类中的属性全部都是内置类型成员,我们可以使用默认生成的拷贝构造函数完成拷贝操作

在这里插入图片描述
我们把Date类自己编写的拷贝构造删除后,尝试使用默认的拷贝构造函数。

class Date
{
public:
	Date() {
		cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}

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

int main() {
	Date d1(2024,1,1);
	printf("d1:>");
	d1.Print();
	
	Date d2(d1);
	printf("d2:>");
	d2.Print();

	return 0;
}

程序运行结果:
在这里插入图片描述
在上图中,我们也可以看出,程序没有编写拷贝构造函数,使用默认拷贝构造函数也完成了任务。但这并不代表可以一直使用默认构造函数
在这里插入图片描述
我们使用stack类来尝试一下,默认拷贝构造函数是否可以完成我们预想的结果。

class stack {
public:
	stack(int defintCapacity = 4) {
		_arr = (int*)calloc(defintCapacity, sizeof(int));
		if (nullptr == _arr)
		{
			perror("malloc申请空间失败");
			return;
		}
		_capacity = defintCapacity;
		_size = 0;
	}

	stack(int* arr, int defintCapacity) {
		if (nullptr == arr) {
			perror("malloc申请空间失败");
			return;
		}
		_arr = arr;
		_capacity = defintCapacity;
		_size = 0;
	}

	void push(int x) {
		//....扩容等
		_arr[_size++] = x;
	}
	 
	~stack() {
		free(_arr);
		_arr = nullptr;
	}
private:
	int* _arr;
	int _size;
	int _capacity;
};

int main() {
	stack s1;

	stack s2(s1);

	return 0;
}

程序运行结果:

在这里插入图片描述

  • 因为在默认的拷贝构造函数对象按内存存储按字节序完成拷贝,在拷贝结束后发现 s2对象的_arr数组地址和s1对象的_arr数组地址一样
  • 这时候我们如果我们s2存储内容,也会改变s1对象的内容,这不是我们想要的,而且对象结束生命周期之后对象会自动调用自己对应的析构函数。这时候析构函数多次释放同一个空间程序崩溃,如下图。
    在这里插入图片描述

使用浅拷贝的内存图布局如下图:
在这里插入图片描述
要解决这种问题,就需要涉及深拷贝(不才后面会专门写一篇笔记),下面代码是针对这种情况的特殊解决办法。

class stack {
public:
	stack(int defintCapacity = 4) {
		_arr = (int*)calloc(defintCapacity, sizeof(int));
		if (nullptr == _arr)
		{
			perror("malloc申请空间失败");
			return;
		}
		_capacity = defintCapacity;
		_size = 0;
	}

	stack(int* arr, int defintCapacity) {
		if (nullptr == arr) {
			perror("malloc申请空间失败");
			return;
		}
		_arr = arr;
		_capacity = defintCapacity;
		_size = 0;
	}
	stack(stack& s) { //需要程序猿自己编写拷贝构造函数(深拷贝)
		_arr = (int*)calloc(s._capacity, sizeof(int));
		if (nullptr == _arr)
		{
			perror("malloc申请空间失败");
			return;
		}
		memcpy(_arr, s._arr, s._capacity * sizeof(int));
		_capacity = s._capacity;
		_size = s._size;
	}

	void push(int x) {
		//....扩容等
		_arr[_size++] = x;
	}
	 
	~stack() {
		free(_arr);
		_arr = nullptr;
	}
private:
	int* _arr;
	int _size;
	int _capacity;
};

我们解决了内置类型拷贝构造的问题后,我们自定义类型是否需要每个都写拷贝构造函数呢?

在这里插入图片描述
我们使用栈实现队列

class stack {
public:
	stack(int defintCapacity = 4) {
		_arr = (int*)calloc(defintCapacity, sizeof(int));
		if (nullptr == _arr)
		{
			perror("malloc申请空间失败");
			return;
		}
		_capacity = defintCapacity;
		_size = 0;
	}

	stack(int* arr, int defintCapacity) {
		if (nullptr == arr) {
			perror("malloc申请空间失败");
			return;
		}
		_arr = arr;
		_capacity = defintCapacity;
		_size = 0;
	}

	void push(int x) {
		//....扩容等
		_arr[_size++] = x;
	}
	 
	~stack() {
		free(_arr);
		_arr = nullptr;
	}

	stack(stack& s) {
		_arr = (int*)calloc(s._capacity, sizeof(int));
		if (nullptr == _arr)
		{
			perror("malloc申请空间失败");
			return;
		}
		memcpy(_arr, s._arr, s._capacity * sizeof(int));
		_capacity = s._capacity;
		_size = s._size;
	}
private:
	int* _arr;
	int _size;
	int _capacity;
};

class MyQueue{

private:
	stack s1;
	stack s2;
};

int main() {
	stack s1;
	s1.push(1);
	s1.push(2);
	s1.push(3);

	stack s2(s1);

	return 0;
}

程序运行结果:
在这里插入图片描述

  • 这时,MyQueue类就不需要编写拷贝构造,因为在MyQueue对象进行拷贝时,会自动调用stack类的拷贝构造。
  • 不写拷贝构造函数不需要写构造函数与析构函数的逻辑是一样的。
  • 拷贝构造函数形参也尽可能的加上const修饰权限范围,这样可以防止形参对象的属性被修改,而且也可以防止实参的权限放大的问题

二、运算符重载

C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,其中在自定义类型中,所有的运算符都需要程序员进行运算符重载
在这里插入图片描述
日期类比较大小

class Date
{
public:
	Date() {
		cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}

public:
	int _year = 1;
	int _month = 1;
	int _day = 1;
};

bool Less(const Date& s1, const Date& s2) {
	if (s1._year < s2._year) {
		return true;
	}
	else if (s1._year == s2._year && s1._month < s2._month) {
		return true;
	}
	else if (s1._year == s2._year && s1._month == s2._month && s1._day < s2._day) {
		return true;
	}

	return false;
}

bool Larger(const Date& s1, const Date& s2) {
	if (s1._year > s2._year) {
		return true;
	}
	else if (s1._year == s2._year && s1._month > s2._month) {
		return true;
	}
	else if (s1._year == s2._year && s1._month == s2._month && s1._day > s2._day) {
		return true;
	}

	return false;
}
int main() {
	Date d1(2022,5,4);	
	Date d2(2022,5,5);
	
	cout << Less(d1, d2) << endl;
	cout << Larger(d1, d2) << endl;

	return 0;
}
  • 在我们自定义的日期类中,我们想比较两个类的大小只能通过函数的形式来比较
  • 如果函数命名不规范时,我们难以比较

所以在C++中增加了赋值运算符重载,方便程序猿的使用

运算符重载关键字:operator后面接需要重载的运算符符号

我们使用运算符重载来改善上述代码:

class Date
{
public:
	Date() {
		cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}

public:
	int _year = 1;
	int _month = 1;
	int _day = 1;
};

bool operator<(const Date& s1, const Date& s2) {
	if (s1._year < s2._year) {
		return true;
	}
	else if (s1._year == s2._year && s1._month < s2._month) {
		return true;
	}
	else if (s1._year == s2._year && s1._month == s2._month && s1._day < s2._day) {
		return true;
	}

	return false;
}

bool operator>(const Date& s1, const Date& s2) {
	if (s1._year > s2._year) {
		return true;
	}
	else if (s1._year == s2._year && s1._month > s2._month) {
		return true;
	}
	else if (s1._year == s2._year && s1._month == s2._month && s1._day > s2._day) {
		return true;
	}

	return false;
}
int main() {
	Date d1(2022, 5, 4);
	Date d2(2022, 5, 5);

	cout << operator<(d1, d2) << endl;
	cout << operator>(d1, d2) << endl;

	return 0;
}

我们只需要把函数名修改为:operator<operator>。这样就完成了 <>的运算符重载。我们在main函数中调用也是如此,使用operator<operator>来调用此函数。但是这样就和我们编写函数没什么区别。

运算符重载的作用是可以直接在main函数中使用<> 来进行比较。如下:

int main() {
	Date d1(2022, 5, 4);
	Date d2(2022, 5, 5);

	cout << operator<(d1, d2) << endl;
	cout << operator>(d1, d2) << endl;

	printf("\n");

	cout << (d1 < d2) << endl;
	cout << (d1 > d2) << endl;
	return 0;
}

程序运行结果:

在这里插入图片描述

  • 虽然我们是使用了<>来进行比较,但是我们通过operater来重载<>后,我们就可以直接使用<>来进行比较。本质上还是使用operator<operator>来调用此函数编译器会自己处理的过程。如下图:在这里插入图片描述

  • 这时,operater重载的<>是全局的,对类中的属性要求就一定是public类型,如果是私有的无法访问,所以我们要重载类对象的<>时,可以把operater函数作为类的成员函数。

运算符重载必须使用满足C++对运算符重载的规定:

  1. 不能通过连接其他符号来创建新的操作符:比如operator@
  2. 重载操作符必须有一个类类型参数,即不能全部是内置类型,比如bool operator+(int& a ,int& b){...}
  3. 用于内置类型的运算符,其含义不能改变,例如:内置的整型+,不能改变其含义,如改为减…
  4. 作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐藏的this
  5. .* ::(域作用限定符) sizeof ?:(三目) . 注意以上5个运算符不能重载。
    在这里插入图片描述
    我们把上述例子的全局运算符重载函数改为Date类的内置成员函数。
class Date
{
public:
	Date() {
		cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}

	bool operator<(const Date& s2) {
		if (_year < s2._year) {
			return true;
		}
		else if (_year == s2._year && _month < s2._month) {
			return true;
		}
		else if (_year == s2._year && _month == s2._month && _day < s2._day) {
			return true;
		}

		return false;
	}

	bool operator>( const Date& s2) {
		if (_year > s2._year) {
			return true;
		}
		else if (_year == s2._year && _month > s2._month) {
			return true;
		}
		else if (_year == s2._year && _month == s2._month && _day > s2._day) {
			return true;
		}

		return false;
	}
public:
	int _year = 1;
	int _month = 1;
	int _day = 1;
};


int main() {
	Date d1(2022, 5, 4);
	Date d2(2022, 5, 5);

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

	printf("\n");

	cout << d1.operator<(d2) << endl;
	cout << d1.operator>(d2) << endl;

	return 0;
}
  • 作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐藏的this
  • 而且作为类成员函数重载时,就算属性是私有的,我们也可以进行比较。
  • 类成员函数重载时,在main函数中,也可以直接使用<>来进行比较,编译器最终也是会转换为d1.operator<(d2)在这里插入图片描述

2.1、赋值运算符重载(类的默认成员函数)

赋值运算符重载与 拷贝构造函数不同处:

  • 拷贝构造函数适用于:用一个已经存在的对象初始化另外一个对象
  • 赋值运算符重载函数适用于:已经存在的两个对象之间赋值拷贝

在这里插入图片描述

class Date
{
public:
	Date() {
		cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}
	
	Date(const Date& d1) {
	cout << "Date(const Date& d1)" << endl;
		_year = d1._year;
		_month = d1._month;
		_day = d1._day;
	}

	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}

	bool operator<(const Date& s2) {
		if (_year < s2._year) {
			return true;
		}
		else if (_year == s2._year && _month < s2._month) {
			return true;
		}
		else if (_year == s2._year && _month == s2._month && _day < s2._day) {
			return true;
		}

		return false;
	}

	bool operator>(const Date& s2) {
		if (_year > s2._year) {
			return true;
		}
		else if (_year == s2._year && _month > s2._month) {
			return true;
		}
		else if (_year == s2._year && _month == s2._month && _day > s2._day) {
			return true;
		}

		return false;
	}
	
	void operator=(const Date& s2) {
		_year = s2._year;
		_month = s2._month;
		_day = s2._day;
	}
public:
	int _year = 1;
	int _month = 1;
	int _day = 1;
};


int main() {
	Date d1(2022, 5, 4);
	Date d2(2000, 1, 1);

	d1 = d2;

	return 0;
}

程序运行结果

在这里插入图片描述

  • 程序运行结果没有问题,但是我们在内置类型赋值的时候可以连续赋值的,如int a,b; a=b=12;,但是我们把赋值运算符重载的返回值设置了void,程序运行不了连续赋值的结果。如下图:在这里插入图片描述

我们可以把返回值设置为Date类作为返回值。如下程序:

Date operator=(const Date& s2) {
	_year = s2._year;
	_month = s2._month;
	_day = s2._day;
	
	return *this;
}
  • 这样我们就可以完成连续赋值的处理,如下图
    在这里插入图片描述

但是,我们直接使用传值返回,会造成大量无用的拷贝构造,这样会对性能造成一点影响:在这里插入图片描述

我们就可以把传值返回改为传引用返回虽然this指针是形参,会随着函数的生命周期结束而销毁,但是我们可以*this作为返回,这样我们返回的是对象的地址对象的生命周期不会随着函数的生命周期结束而销毁。

Date& operator=(const Date& s2) {
	_year = s2._year;
	_month = s2._month;
	_day = s2._day;
	
	return *this;
}

程序运行结果:
在这里插入图片描述

  • d1 = d2其底层与d1.operator=(d2)是一样的在这里插入图片描述
  • 当然,在使用重载赋值运算符时,程序猿会出现自己与自己赋值的情况。这时候我们可以在重载运算符中加入自己给自己赋值的条件判断。这样可以避免自己给自己赋值。
Date& operator=(const Date& s2) {
	if (this != &s2) {
		_year = s2._year;
		_month = s2._month;
		_day = s2._day;
	}
		
	return *this;
}

赋值运算符重载格式:

  • 参数类型: const T&,传递引用可以提高传参效率
  • **返回值类型:**T&,返回引用可以提高返回的效率,有返回值目的是为了支持连续赋值
  • 检测是否自己给自己赋值
  • 返回*this 要复合连续赋值的含义

因为赋值运算符重载是类的默认成员函数,即用户没有显式实现时,编译器会生成一个默认赋值运算符重载,以值的方式逐字节拷贝。

注意编译器默认生成的赋值运算符重载函数:内置类型成员变量是直接赋值的,而自定义类型成员变量需要调用对应类的赋值运算符重载完成赋值。与拷贝构造的行为一样。

需要注意的是,编译器生成的赋值运算符拷贝逐字节拷贝即浅拷贝的方式。 遇到指针等相关拷贝的情况,需要程序猿编写深拷贝。

赋值运算符只能重载成类的成员函数不能重载成全局函数

原因:赋值运算符如果不显式实现,编译器会生成一个默认的。此时用户再在类外自己实现一个全局的赋值运算符重载,就和编译器在类中生成的默认赋值运算符重载冲突了,故赋值运算符重载只能是类的成员函数。
在这里插入图片描述

2.2、重载运算符中的特殊:前置++和后置++的重载

在重载运算符中有前置++后置++两个特殊的运算符重载。因为在程序中这两个运算符的函数名都是一样的。需要函数重载来区别前置++后置++。(重载运算符--同理)
在这里插入图片描述
我们先实现前置++,还是使用上述的Date类为例。

class Date
{
public:
	Date() {
		//cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}
	Date(const Date& d1) {
		cout << "Date(const Date& d1)" << endl;
		_year = d1._year;
		_month = d1._month;
		_day = d1._day;
	}

	void Print(){
		cout << _year << "-" << _month << "-" << _day << endl;
	}

	bool operator<(const Date& s2) {
		if (_year < s2._year) {
			return true;
		}
		else if (_year == s2._year && _month < s2._month) {
			return true;
		}
		else if (_year == s2._year && _month == s2._month && _day < s2._day) {
			return true;
		}

		return false;
	}

	bool operator>(const Date& s2) {
		if (_year > s2._year) {
			return true;
		}
		else if (_year == s2._year && _month > s2._month) {
			return true;
		}
		else if (_year == s2._year && _month == s2._month && _day > s2._day) {
			return true;
		}

		return false;
	}
	
	Date& operator=(const Date& s2) {
		_year = s2._year;
		_month = s2._month;
		_day = s2._day;
		
		return *this;
	}

	Date& operator++() {//前置++
		_day++;
		return *this;//因为是前置++,可以把加完后的结果直接当做返回值。
	}

public:
	int _year = 1;
	int _month = 1;
	int _day = 1;
};

int main() {
	Date d1(2022, 5, 4);
	
	++d1;
	d1.Print();
	return 0;
}

程序运行结果:

在这里插入图片描述

  • operator++():因为++是自增不需要参数,所以我们把形参列表设为空。

我们operator++()当做为前置++,但后置++也是相同的函数名operator++,所以在C++中我们后置++的形参中增加一个参数,让其形成函数重载,以达到前置++后置++的区别

但是 ++是自增,不需要接收任何形参,但是需要完成函数重载,所以我们只需要在形参列表中增加一个int类型即可完成重载,该类型可以不添加变量名称。如下

	Date& operator++() {//前置++
		_day++;
		return *this;//因为是前置++,可以把加完后的结果直接当做返回值。
	}
	
	Date operator++(int) {//后置++
		Date d1 = *this;//因为是后置加加,所以要把原来的结果先储存为一个临时变量。之后,我们才可以进行自增处理
		_day++;
		return d1;//返回的是临时变量的值。
	}
  • 后置++中,operator++(int)int参数不是为了接收具体的值。仅仅是占位,跟前置++构成函数重载。

其中前置++后置++的调用会由编译器区分。我们程序员不需要理会如何进行调用。

在我们程序员自己编写的前置++后置++会有性能的区别。因为 后置++需要进行拷贝构造,创建一个临时变量来作为返回值,而且也是传值返回。但是在内置类型的++--中性能区别不大

2.3、流插入与流提取的重载运算符

在上面例中,我们Date类对象的内容打印是使用print函数来进行Date类属性的显示,为了更好的打印,我们可以使用流的重载运算符来进行自定义类型的打印。
在这里插入图片描述
不才在文档查询网站中,查询到cinistream类的对象和coutosterem类的对象,那么我们在Date类中,就可以通过istream类和osterem类来自定义引用对象来进行标准输入输出。
在这里插入图片描述
我们先以打印为例,即重载<<运算符。

class Date
{
public:
	Date() {
		//cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}
	Date(const Date& d1) {
		cout << "Date(const Date& d1)" << endl;
		_year = d1._year;
		_month = d1._month;
		_day = d1._day;
	}
	void operator<<(ostream& out) {
		out << _year << "年" << _month << "月" << _day << "日";
	}

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


int main() {
	Date d1(2022, 5, 4);
	//d1.Print();
	cout << d1;
	Date n1 = d1;
	//n1.Print();
	cout << d1 << endl << n1 << endl;

	return 0;
}

运行结果:在这里插入图片描述

  • 因为我们是把重载运算符写入了类中,成为了类的方法,所以我们直接使用cout << d1是无法访问的
  • cout << d1其本质是cout.operator(d1),我们在库中是没有自定义类型重载的。如下图在这里插入图片描述
  • 所以我们在Date类中重载的<<运算符,需要d1 << cout这样来使用,因为这才符合d1.operator(cout)的函数调用

但是d1.operator(cout)明显不符合我们是使用习惯,但是在类中,第一个形参默认是this指针,无法改变的。所以我们就把operator<<定义在类外面,作为全局函数。

作为全局重载运算符函数时,我们久可以把第一个参数设置为ostrem,第二次参数设置为我们自定义类型Date,这样久可以完成cout << d1的使用。如下图
在这里插入图片描述

  • 此时虽然,cout << d1没有报错,但是我们类中的属性时私有的,为了访问到我们私有的属性,我们在类中可以把全局重载运算符函数operater<<声明为友元函数。(不才后面会专门出一篇笔记讲解)
class Date
{
	friend void operator<<(ostream& out, const Date& d1);//把operator<<函数声明为友元函数
	
public:
	Date() {
		//cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}
	Date(const Date& d1) {
		//cout << "Date(const Date& d1)" << endl;
		_year = d1._year;
		_month = d1._month;
		_day = d1._day;
	}

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

void operator<<(ostream& out, const Date& d1) {
	out << d1._year << "年" << d1._month << "月" << d1._day << "日";
}
  • 在类中声明前加上friend关键字,声明void operator<<(ostream& out, const Date& d1)函数时友元函数。

运行结果:
在这里插入图片描述
和之前的运算符重载函数相同,我们为了多次运用<<,我们的返回值久设置为ostrem,这样我们就可以实现复合使用cout << d1 << endl。又因为coutiostream创建的对象,而且cout的生命周期是程序的声明周期,所以我们可以把ostream设置为ostream&引用返回

ostream& operator<<(ostream& out, const Date& d1) {
	out << d1._year << "年" << d1._month << "月" << d1._day << "日";
	return out;
}

运行结果:
在这里插入图片描述
同理,流插入也是如此

istream& operator>>(istream& in, Date& d1) {
	in >> d1._year >> d1._month >> d1._day;
	return in;
}
  • 也需要在类中声明为友元函数。

2.4、重载取地址操作符(是类的默认成员)

在这里插入图片描述
因为也只是取地址,所以函数设计就可以非常简单。只需要返回值是:类类型指针。返回值是this指针即可

Date* operator&() {
	cout << "Date* operator&()" << endl;
	return this;
}

如果遇到const修饰的话,我们也需要this指针设置为const成员,而且返回值也是const+类类型指针。如下程序:

const Date* operator&() const {
		cout << "Date* operator&() const" << endl;
		return this;
	}

编译器会根据是否为const修饰的对象调用对应的取地址运算符重载函数

class Date
{
public:
	Date() {
		//cout << "Date()" << endl;
	}

	Date(int year, int month, int day) {
		_year = year;
		_month = month;
		_day = day;
	}
	Date(const Date& d1) {
		//cout << "Date(const Date& d1)" << endl;
		_year = d1._year;
		_month = d1._month;
		_day = d1._day;
	}
	Date* operator&() {
		cout << "Date* operator&()" << endl;
		return this;
	}

	const Date* operator&() const {
		cout << "Date* operator&() const" << endl;
		return this;
	}

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

int main() {
	Date d1(2022, 5, 4);

	const Date d2(200, 1, 1);

	Date* pd1 = &d1;

	const Date* pd2 = &d2;

	return 0;
}

程序运行结果为:

在这里插入图片描述
重载取地址运算符并没有什么实际上的作用。因为属于类的默认成员函数,不需要程序员写会默认生成。所达到的效果是一样的。


以上就是本章所有内容。若有勘误请私信不才。万分感激💖💖 如果对大家有用的话,就请多多为我点赞收藏吧~~~💖💖
请添加图片描述

ps:表情包来自网络,侵删🌹

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

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

相关文章

Linux:SystemV通信

目录 一、System V通信 二、共享内存 代码板块 总结 三、信号量 信号量理论 信号量接口 一、System V通信 System V IPC&#xff08;inter-process communication&#xff09;&#xff0c;是一种进程间通信方式。其实现的方法有共享内存、消息队列、信号量这三种机制。 …

2025.1.15——七、cookie注入

题目来源&#xff1a;ctfhub技能树 目录 一、打开靶机&#xff0c;整理已知信息 二、解题步骤 step 1&#xff1a;按F12查看cookie信息&#xff0c;见上 step 2&#xff1a;bp抓包修改cookie信息&#xff0c;确认注入类型 step 3&#xff1a;查看字段数 step 4&#xff…

【简博士统计学习方法】第2章:3. 感知机——学习算法之原始形式:算法解说

3. 感知机——学习算法之原始形式&#xff1a;算法解说 3.1 学习问题 给定训练数据集&#xff1a; T { ( x 1 , y 1 ) , ( x 2 , y 2 ) ⋯ , ( x N , y N ) } T\left\{\left(x_{1}, y_{1}\right),\left(x_{2}, y_{2}\right) \cdots,\left(x_{N}, y_{N}\right)\right\} T{(x…

【Flink系列】10. Flink SQL

10. Flink SQL Table API和SQL是最上层的API&#xff0c;在Flink中这两种API被集成在一起&#xff0c;SQL执行的对象也是Flink中的表&#xff08;Table&#xff09;&#xff0c;所以我们一般会认为它们是一体的。Flink是批流统一的处理框架&#xff0c;无论是批处理&#xff08…

web漏洞扫描有什么作用?web漏洞扫描原理

Web漏洞扫描在网络安全领域中扮演着至关重要的角色&#xff0c;web漏洞扫描有什么作用&#xff1f;Web漏洞扫描能够自动化地检测Web应用中的潜在安全漏洞&#xff0c;包括但不限于SQL注入、跨站脚本&#xff08;XSS&#xff09;、跨站请求伪造&#xff08;CSRF&#xff09;、文…

【Idea启动项目报错NegativeArraySizeException】

项目场景&#xff1a; Idea启动项目报错&#xff08;打包不报错&#xff09;&#xff0c;项目在服务器部署运行没有问题&#xff0c;尝试了重启idea、重启电脑、maven clean/install 都不行 maven-resources-production:sample: java.lang.NegativeArraySizeException: -5833…

小程序组件 —— 31 事件系统 - 事件绑定和事件对象

小程序中绑定事件和网页开发中绑定事件几乎一致&#xff0c;只不过在小程序不能通过 on 的方式绑定事件&#xff0c;也没有 click 等事件&#xff0c;小程序中绑定事件使用 bind 方法&#xff0c;click 事件也需要使用 tap 事件来进行代替&#xff0c;绑定事件的方式有两种&…

sparkSQL练习

1.前期准备 &#xff08;1&#xff09;建议先把这两篇文章都看一下吧&#xff0c;然后把这个项目也搞下来 &#xff08;2&#xff09;看看这个任务 &#xff08;3&#xff09;score.txt student_id,course_code,score 108,3-105,99 105,3-105,88 107,3-105,77 105,3-245,87 1…

GIFT ICA 下载记录

1.帮助文档 Group ICA/IVA Of fMRI Toolbox&#xff1b;【GIFT介绍】 Group ICA of fMRI Toolbox (GIFT) Walk Through&#xff1b;【流程介绍】 GIFT v1.3c Functions Srinivas Rachakonda, Eric Egolf and Vince Calhoun【流程解释】 2.下载记录 从官网下载程序包&#xff0…

从零深度学习:(2)最小二乘法

今天我们从比较简单的线性回归开始讲起&#xff0c;还是一样我们先导入包 import numpy as np import torch import matplotlib as mpl import matplotlib.pyplot as plt a torch.arange(1,5).reshape(2,2).float() a 我们利用刚刚导入的画图的包将这两个点画出来&#xff0…

02JavaWeb——JavaScript-Vue(项目实战)

一、JavaScript html完成了架子&#xff0c;css做了美化&#xff0c;但是网页是死的&#xff0c;我们需要给他注入灵魂&#xff0c;所以接下来我们需要学习 JavaScript&#xff0c;这门语言会让我们的页面能够和用户进行交互。 1.1 介绍 通过JS/js效果演示提供资料进行效果演…

【Flink系列】5. DataStream API

5. DataStream API DataStream API是Flink的核心层API。一个Flink程序&#xff0c;其实就是对DataStream的各种转换。具体来说&#xff0c;代码基本上都由以下几部分构成&#xff1a; 5.1 执行环境&#xff08;Execution Environment&#xff09; Flink程序可以在各种上下文…

大模型高并发部署方案探究

版本 内容 姓名 时间 V1.0 新建 xx 2025-01-16 声明&#xff1a;只是进行探究&#xff0c;后续真正实践后&#xff0c;会更新新的内容 前置条件&#xff1a;70B的模型&#xff0c;并发要求200 性能测试参考链接 Benchmarking LLM Inference Backends :表明一台A100(8…

MIAOYUN信创云原生项目亮相西部“中试”生态对接活动

近日&#xff0c;以“构建‘中试’生态&#xff0c;赋能科技成果转化”为主题的“科创天府智汇蓉城”西部“中试”生态对接活动在成都高新区菁蓉汇隆重开幕。活动分为成果展览、“中试”生态主场以及成果路演洽谈对接三大板块。在成果展览环节&#xff0c;成都元来云志科技有限…

pytest-instafail:让测试失败信息即时反馈

pytest-instafail&#xff1a;让测试失败信息即时反馈 前言一、简介二、优势三、安装与使用3.1 未安装时运行情况3.2 安装3.3 已安装时运行情况3.3 pytest.ini 配置选项 四、对比 总结 前言 当测试用例数量庞大时&#xff0c;定位测试失败的原因往往耗时费力。此时&#xff0c;…

低代码平台:技术复杂性的系统简化

在传统开发模式下&#xff0c;应用构建需要经历需求分析、代码开发、测试部署等多环节&#xff0c;流程繁琐且耗时&#xff0c;往往成为企业技术创新的瓶颈。低代码平台通过模块化和自动化技术重新定义开发流程&#xff0c;使开发者能够在较短时间内实现复杂的应用功能&#xf…

精度论文:【Focaler-IoU: More Focused Intersection over Union Loss】

Focaler-IoU: 更聚焦的交并比损失 Focaler-IoU: More Focused Intersection over Union Loss Focaler-IoU: 更聚焦的交并比损失I. 引言II. 相关工作III. 方法IV. 实验V. 结论 原文地址&#xff1a;官方论文地址 代码地址&#xff1a;官方代码地址 摘要——边界框回归在目标检…

“AI智慧化服务系统:未来生活的智能管家

在当今快速发展的科技时代&#xff0c;人工智能&#xff08;AI&#xff09;正以前所未有的速度改变着我们的生活。AI智慧化服务系统作为这一变革的前沿技术&#xff0c;正在逐渐成为我们未来生活的智能管家。它们不仅提高了服务效率&#xff0c;还为我们带来了更加个性化和便捷…

nginx 修改内置 404 页面、点击劫持攻击。

1、在部署前端项目的目录下增加 404.html 页面&#xff1a;/opt/web/404.html。 2、在 nginx 配置中增加 404 配置&#xff1a; root /opt/web; # 设置根目录的配置error_page 404 404.html; location /404.html {root /opt/web;# 指定 404 页面所在的根目录internal;# 确保…

网络密集型应用的Linux网络缓冲区参数优化

一、网络IO密集型 1.哪些应用属于网络IO密集型应用 文件上传、下载服务器&#xff0c;实时大数据同步复制&#xff0c;Kafka巨量数据QPS生产消费环境&#xff0c;CDN等环境都是网络IO密集型的服务应用 2.知识来源 在《kafka权威指南2》书中环境搭建的网络小节写到了几个参数…