C++笔记 06 (运算符重载)

一. 运算符重载

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

1. 加号运算符重载

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

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

		//成员函数实现加号运算符重载
		Person operator+ (const Person &p)
		{
			Person temp;
			temp.m_A=this->m_A+p.m_A;
			temp.m_B=this->m_B+p.m_B;
			return temp;
		}
	public:
		int m_A;
		int m_B;
};

//全局函数实现加号运算符重载
Person operator+ (const Person &p1 ,const Person &p2)
	{
		Person temp(0,0);
		temp.m_A=p1.m_A+p2.m_A;
		temp.m_B=p1.m_B+p2.m_B;
		return temp;
	}

//运算符重载可以发生函数重载
Person operator+ (const Person p2,int val)
{
	Person temp;
	temp.m_A=p2.m_A+val;
	temp.m_B=p2.m_B+val;
	return temp;
}

void test()
{
	Person p1(10,10);
	Person p2(20,20);

	// 成员函数方式
	Person p3=p1+p2;//相当于p2.operator+(p1)
	cout<<"m_A:"<<p3.m_A<<"m_B:"<<p3.m_B<<endl;

	Person p4=p3+10; //相当于operator+(p3,10)
	cout<<...<<endl;
}
int main()
{
	... ...
}

总结1: 对于内置的数据类型的表达式的运算符是不可能改变的
总结2:不能滥用运算符重载(例如operator+函数中写成减法)

2. 左移运算符重载

作用:可以输出自定义数据类型

class Person
{
	friend ostream & operator <<(ostream & out ,Person &p);
	public:
		Person(int a,int b)
		{
			this->m_A = a;
			this->m_B = b;
		}
		//成员函数实现不了, p<<cout 不是我们想要的效果
		//void operator << (Person &p) { }
	private: int m_A;
			 int m_B;
};

//全局函数实现左移重载
//ostream对象只能有一个
ostream & operator << (ostream &out,Person &p)
{
	out<<"a:"<<p.m_A<<"b:"<<p.m_B;
	return out;
}

void test()
{
	Person p1(10,20);
	cout<<p1<<"hello"<<endl;
}

int main()
{
	... ...
}

总结:重载左移运算符配合友元可以实现输出自定义数据类型

3. 递增运算符重载

int a=10;
cout<<++a<<endl; //输出11
cout<<a<<endl; //输出11
//前置递增,先进行递增,再输出表达式

int b=10;
cout<<b++<<endl; //输出10
cout<<b<<endl; //输出11
//后置递增,先进行表达式运算,再让变量进行递增操作
//前置递增与后置递增的区别

int main()
{
	int a=10;
	a++; //等价于a=a+1
	cout<<a<<endl; //输出11

	int b=10;
	++b;
	cout<<b<<endl; //输出11

//前置递增先对变量进行++,再计算表达式
int a2=10;
int b2=++a2*10;
cout<<a2<<endl; //输出11
cout<<b2<<endl; //输出110

//后置递增先计算表达式,后对变量进行++
int a3=10;
int b3=a3++*10; //先乘以10,再+1
cout<<a3<<endl; //输出11
cout<<b3<<endl; //输出100
}
class MyInteger
{
	friend ostream& operator <<(ostream& out,MyInteger myint);

	public:
		MyInteger()
		{
			m_Num=0;
		}
		
	//前置++
	MyInteger & operator++() 
	{
		//先++
		m_Num++;
		//再返回
		return *this;
		
		/*此处为什么是返回引用,能不能是值?不能。返回引用是为了
			一直对一个数据进行递增操作*/
	}
	/*若是内置数据类型
	{
		int a=0;
		cout<<++(++a)<<endl; 输出2
		cout<<a<<endl; 输出2
    }*/

	/*若 MyInteger operator ++() {... ...}
	void test()
	{
		MyInteger myint;
		cout<<++(++myint)<<endl; 输出2
		cout<<myint<<endl; 输出1,原因是myint做了一次++后,
		返回的是一个新的变量
	}*/

	//后置++
	MyInteger operator++(int)
	{
       //int代表占位参数,可以用于区分前置和后置递增

	  //先记录当时结果,后递增,最后将记录结果做返回
	  MyInteger temp=*this; 
	  m_Num++;
	  return temp;
	}
	private:
		int m_Num;
};

ostream & operator <<(ostream &out,MyInteger myint)
{
	out<<myint.m_Num;
	return out;
}

//前置++,先++,再返回
void test01()
{
	MyInteger myint;
	cout<<++myInt<<endl;
	cout<<myInt<<endl;
}

后置++,先返回,再++
void test02()
{
	MyInteger myInt;
	cout<<myInt++<<endl;
	cout<<myInt<<endl;
}

int main() {... ...}

总结:前置递增返回引用,后置递增返回值

4. 赋值运算符重载

C++编译器至少给一个类添加4个函数
1)默认构造函数(无参,函数体为空)
2)默认析构函数(无参,函数体为空)
3)默认拷贝构造函数,对属性进行值拷贝
4)赋值运算符operator=,对属性进行值拷贝

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

class Person
{
	public:
		Person (int age)
		{
			//将年龄数据开辟到堆区
			m_Age= new int(age);
		}
		//重载赋值运算符
		Person & operator =(Person &p)
		{
			if(m_Age!=NULL) //先判断是否有属性在堆区,如果有先释放干净,然后再深拷贝
			{
				delete m_Age;
				m_Age=NULL;
			}
			//编译器提供的代码是浅拷贝 即m_Age=p.m_Age;
			//提供深拷贝,解决浅拷贝的问题
			m_Age= new int(*p.m_Age);
			//返回自身
			return *this;
		}
		~Person()
		{
			if(m_Age!=NULL) 
			{
				delete m_Age;
				m_Age=NULL;
			}
		}
		//年龄的指针
		int *m_Age;
}

void test01()
{
	Person p2(20);
	Person p3(30);
	p3=p2=p1; //赋值操作
	cout<<"p1的年龄为:"<<*p1.m_Age<<endl;
	cout<<"p2的年龄为:"<<*p2.m_Age<<endl;
	cout<<"p3的年龄为:"<<*p3.m_Age<<endl;
}
int main()
{... ...}

5. 关系运算符重载

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

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

6. 函数调用运算符重载

1)函数调用运算符()也可以重载
2)由于重载后使用的方式非常向函数的调用,因此称为仿函数
3)仿函数没有固定写法,非常灵活

class Myprint
{
	public:
		void operator()(string test)
		{
			cout<<test<<endl;
		}
};

void test01()
{
	//重载()操作符也称为仿函数
	Myprint myFunc;
	myFunc("hello"); 
}
class Myadd
{
	public:
		int operator()(int v1,int v2)
		{
			return v1+v2;
		}	
};
void test02()
{
	Myadd add;
	int ret=add(10,10)
	cout<<"ret="<<ret<<endl;

	//匿名函数对象
	cout<<Myadd()(100,100)<<endl;
	//创建了一个匿名对象,特点是当前行执行完了。立即被释放
}

int main() {...  ...}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值