学习:类和对象——运算符重载

什么是运算符重载?

运算符重载概念:对已有的运算符重新进行定义,赋予其另一种功能,以适应不同的数据类型

对于个人而言,学到这里的感受就是,多了一种能够自定义方式来进行运算的方式吧,通过类似语法糖的效果operator+来自动实现相应的操作


+号运算符重载:

作用:实现两个自定义数据类型相加的运算

在+号运算符重载里面,我们可以通过两种方式来定义,一种是成员函数来实现,还有一种全局函数来实现

同样的调用方法也有两种,以下代码有体现

示例代码:

#include<iostream>
#include<string>

using namespace std;

class Person{
public:

	//成员函数实现 + 号运算符重载
	//Person operator+(Person& p1) {
	//	Person temp;
	//	temp.age = this->age + p1.age;
	//	return temp;
	//}

public:
	int age;
};

//全局函数 实现+号运算符重载
Person operator+(Person& p1, int val) {
	Person temp;
	temp.age = p1.age + val;
	return temp;
}

int main() {
	Person p1;
	p1.age = 20;
	//Person p2 = p1.operator+(p1); //调用方式有两种,这是第一种
	Person p2 = p1 + 10;//这是第二种调用方式
	cout << p2.age << endl;
	system("pause");
	return 0;
}

总结1:对于内置的数据类型的表达式的的运算符是不可能改变的

总结2:不要滥用运算符重载


左移运算符重载

通过左移运算符重载,我们可以实现输出自定义数据类型,比如 cout << 对象 << endl,实际输出的是 cout << 对象.a << endl

#include<iostream>
#include<string>

using namespace std;

class Person {
	friend ostream& operator<<(ostream& cout, Person& p); //因为a,b是私有成员变量,所以我们利用friend友元进行修饰,让其能够访问到我们的私有变量

public:
	Person(int a, int b) {
		this->a = a;
		this->b = b;
	}

	//void operator<<(cout) {} //在类中定义的话无法实现 cout << p这种形式到时候调用的时候只能是p.operator<<(p),简化版只能是 p << cout,所以我们只在全局区定义函数

private:
	int a;
	int b;
};

ostream& operator<<(ostream& cout,Person& p) {  // 都不需要生成一个新的内存空间,所以都进行传参引用
	cout << p.a << "," << p.b << endl;
	return cout; // 这里返回的是ostream类型 是为了进行后面 << 继续拼接的输出,链式操作
}



int main() {
	Person p1(1, 2);
	cout << p1 << "hello world " <<endl;  //通过p1 直接可以输出对象p1的成员变量
	system("pause");
	return 0;

}

递减运算符重载:

作用: 通过重载递减运算符,实现自己的整型数据

#include<iostream>
#include<string>

using namespace std;

class Num {

	//创建友元函数,因为这里a是私有变量
	friend ostream & operator<<(ostream& cout, Num n);

public:
	Num(int a) {
		this->a = a;
	}

	Num & operator--() {// 定义前置--运算符重载
		a--;
		return *this;
		
	}

	Num operator--(int) { //定义后置--运算符重载 这时候int这种占位参数就体现出来了,区别前置还是后置不要问为什么,因为有个傲娇的编译器
		Num temp = *this;
		this->a--;
		return temp;
	}

private:
	int a;
};

//重载一个左移运算符
ostream & operator<<(ostream& cout, Num n) {
	cout << n.a;
	return cout;
}

int main() {
	Num n1(1);
	cout << --n1 << endl;
	cout << n1-- << endl;
	system("pause");
	return 0;
	
}

赋值运算符重载:

c++编译器至少给一个类添加4个函数

1、默认构造函数(无参,函数体为空)

2、默认析构函数(无参,函数体为空)

3、默认拷贝构造函数,对属性进行值拷贝

4、赋值运算符 operator=, 对属性进行值拷贝

如果类中有属性指向堆区,做赋值操作时也会出现深浅拷贝问题

#include<iostream>
#include<string>

using namespace std;

class Person {
public:
	Person(int height) {
		this->height= new int(height);
	}

	Person & operator=(Person & p) {
		if (this->height != NULL) {
			delete this->height;
			this->height = NULL;
		}
		this->height = new int(*p.height);
		return *this;
	}

	~Person() {
		if (this->height != NULL) {
			delete this->height;
			this->height = NULL;
		}
	}

public:
	int * height;
};

int main() {
	Person p1(180);
	Person p2(170);
	p2 = p1;

	cout << *p1.height << endl;
	cout << *p2.height << endl;
	system("pause");
	return 0;
	
}

关系运算符重载:

作用:重载关系运算符,可以让两个自定义类型对象进行对比操作

还是一样的,就是返回个类型应该来说是个布尔值,因为是==比较

#include<iostream>
#include<string>

using namespace std;

class Person {
public:
	Person(int age) {
		this->age = age;
	
	}

	bool operator==(Person& p) {
		if (this->age == p.age) {
			return true;
		}
		else {
			return false;
		}
	}

	bool operator!=(Person& p) {
		if (this->age == p.age) {
			return false;
		}
		else {
			return true;
		}
	}
public:
	string name;
	int age;
};

int main() {
	Person p1(18);
	Person p2(18);
	if (p1 == p2) {
		cout << "相等" << endl;
	}else{
		cout << "不相等" << endl;
	}

	system("pause");
	return 0;
	
}

函数调用运算符重载:

1、函数调用运算符 () 也可以重载

2、由于重载后使用的方式非常像函数的调用,因此称为仿函数,仿函数没有固定写法,非常灵活

自己总结下这个知识点:

1、第一点是函数调用运算符重载,也就是跟之前的一样加个()就好了
2、第二点是匿名对象类,也可以我们后面不需要再继续用到,所以就不需要进行储存操作,一次性调用就可以了

体现代码:

#include<iostream>
#include<string>

using namespace std;

class Printer{
public:
	void operator()(string name) {
		cout << name << endl;
	}
};


int main() {
	Printer p1;
	p1("adexx"); //一种方式来调用

	Printer()("adexx");//使用匿名类来进行调用
	system("pause");
	return 0;
	
}
posted @ 2019-11-15 13:04  zpchcbd  阅读(427)  评论(0编辑  收藏  举报