C++ Primer Plus学习笔记之运算符重载

C++ Primer Plus学习笔记之运算符重载

1,成员函数和友元函数选择的建议
下面我们先看两个例子:
成员函数重载
#include<iostream>

using namespace std;

class Complex
{
	public:
		Complex(double r=0,double i=0)
		{
			re=r;
			im=i;
		}
		Complex operator+(const Complex& obj);
		Complex operator!();
		void Display();
	private:
		double re;
		double im;
};

Complex Complex::operator+(const Complex& obj)
{
	Complex temp;
	temp.re=re+obj.re;
	temp.im=im+obj.im;
	return temp;
}

Complex Complex::operator!()
{
	Complex temp;
	temp.re=re;
	temp.im=-im;
	return temp;
}

void Complex::Display()
{
	cout<<re;
	if(im>0)
		cout<<"+"<<im<<"i"<<endl;
	else
		cout<<im<<"i"<<endl;
}

int main(int argc,char *argv[])
{
	Complex obj1(1,2),obj2(3,4);
	Complex obj3=obj1+!obj2;
	obj3.Display();

	obj3=obj1+27;
	obj3.Display();

	//obj3=27+obj1;
	//obj3.Display(); 错误,27不是Complex的对象

	return 0;
}
友元函数重载
#include<iostream>

using namespace std;

class Complex
{
	public:
		Complex(double r=0,double i=0)
		{
			re=r;
			im=i;
		}
		friend Complex operator+(const Complex& obj1,const Complex& obj2);
		Complex operator!();
		void Display();
	private:
		double re;
		double im;
};

Complex operator+(const Complex& obj1,const Complex& obj2)
{
	Complex temp;
	temp.re=obj1.re+obj2.re;
	temp.im=obj1.im+obj2.im;
	return temp;
}

Complex Complex::operator!()
{
	Complex temp;
	temp.re=re;
	temp.im=-im;
	return temp;
}

void Complex::Display()
{
	cout<<re;
	if(im>0)
		cout<<"+"<<im<<"i"<<endl;
	else
		cout<<im<<"i"<<endl;
}

int main(int argc,char *argv[])
{
	Complex obj1(1,2),obj2(3,4);
	Complex obj3=obj1+!obj2;
	obj3.Display();

	obj3=obj1+27;
	obj3.Display();

	obj3=27+obj1;
	obj3.Display();

	return 0;
}
对于成员函数重载,由于obj3=27+obj1的显示调用为obj3=27.operator+(obj1),而27不是Complex类型,因此编译器不能调用Complex的operator+()重载运算符完成加法操作;
对于友元函数重载,由于obj3=27+obj1的调用通过有一个参数缺省的构造函数Complex(double r=0,double i=0),系统可以自动将27转换为Complex(27)对象,因此编译通过;
由此可知,成员函数仅仅能为一个实际对象所调用,友元无此限制。因此若运算符的操作需要修改类对象的状态,则它应该是成员函数,而不应该是友元。相反,如果运算符的操作数(尤其是第一个操作数)希望有隐式类型转换,则该运算符重载必须用友元函数,而不是成员函数。
2,重载++的前缀和后缀
成员函数重载
#include<iostream>

using namespace std;

class Complex
{
	public:
		Complex(double r=0,double i=0)
		{
			re=r;
			im=i;
		}
		friend Complex operator+(const Complex& obj1,const Complex& obj2);
		Complex& operator++();//前缀方式
		Complex operator++(int);//后缀方式
		Complex operator!();
		void Display();
	private:
		double re;
		double im;
};

Complex operator+(const Complex& obj1,const Complex& obj2)
{
	Complex temp;
	temp.re=obj1.re+obj2.re;
	temp.im=obj1.im+obj2.im;
	return temp;
}

Complex& Complex::operator++()
{
	re++;
	im++;
	return *this;
}

Complex Complex::operator++(int)//int为占位符
{
	Complex temp(*this);//拷贝构造函数
	re++;
	im++;
	return temp;
}

Complex Complex::operator!()
{
	Complex temp;
	temp.re=re;
	temp.im=-im;
	return temp;
}

void Complex::Display()
{
	cout<<re;
	if(im>0)
		cout<<"+"<<im<<"i"<<endl;
	else
		cout<<im<<"i"<<endl;
}

int main(int argc,char *argv[])
{
	Complex obj1(1,2),obj2;
	cout<<"First obj1=";
	obj1.Display();

	obj2=obj1++;
	cout<<"Second obj1=";
	obj1.Display();
	cout<<"and obj2=";
	obj2.Display();

	obj2=++obj1;
	cout<<"Third obj1=";
	obj1.Display();
	cout<<"and obj2=";
	obj2.Display();

	return 0;
}
友元函数重载
#include<iostream>

using namespace std;

class Complex
{
	public:
		Complex(double r=0,double i=0)
		{
			re=r;
			im=i;
		}
		friend Complex operator+(const Complex& obj1,const Complex& obj2);
		friend Complex& operator++(Complex& obj);//前缀方式
		friend Complex operator++(Complex& obj,int);//后缀方式
		Complex operator!();
		void Display();
	private:
		double re;
		double im;
};

Complex operator+(const Complex& obj1,const Complex& obj2)
{
	Complex temp;
	temp.re=obj1.re+obj2.re;
	temp.im=obj1.im+obj2.im;
	return temp;
}

Complex& operator++(Complex& obj)
{
	obj.re++;
	obj.im++;
	return obj;
}

Complex operator++(Complex& obj,int)//int为占位符
{
	Complex temp(obj);//拷贝构造函数
	obj.re++;
	obj.im++;
	return temp;
}

Complex Complex::operator!()
{
	Complex temp;
	temp.re=re;
	temp.im=-im;
	return temp;
}

void Complex::Display()
{
	cout<<re;
	if(im>0)
		cout<<"+"<<im<<"i"<<endl;
	else
		cout<<im<<"i"<<endl;
}

int main(int argc,char *argv[])
{
	Complex obj1(1,2),obj2;
	cout<<"First obj1=";
	obj1.Display();

	obj2=obj1++;
	cout<<"Second obj1=";
	obj1.Display();
	cout<<"and obj2=";
	obj2.Display();

	obj2=++obj1;
	cout<<"Third obj1=";
	obj1.Display();
	cout<<"and obj2=";
	obj2.Display();

	return 0;
}
由于++运算符修改类对象本身,因此在友元调用方式中,为了能够修改传递进函数的参数,以对象引用的方式传递。++的前缀方式返回类型为类对象的引用,后缀方式返回类型是类对象,是因为前缀方式返回的是变化之后的类对象,该类对象是作为对象引用传递进函数的,在函数调用结束之后不会被销毁;而后缀方式,函数的返回值是参数对象被修改之前的值(放到局部对象中暂时存储),因此返回的是局部对象的值,而局部对象在函数返回的时候会被销毁,因此函数的返回值只能将局部对象的值拷贝过来,而不能直接使用局部对象空间。故后缀方式返回类型为类对象,而不能是类对象的引用。
3,重载赋值运算符
#include<iostream>
#include<cstring>
#include<cassert>

using namespace std;

class String
{
	public:
		String()
		{
			msize=0;
			mstr=NULL;
		}
		String(char *str);
		String(const String& str);
		String operator+(const String& str);
		String& operator=(const String& str);
		void Display();
	private:
		char *mstr;
		int msize;
};

String::String(char *str)
{
	if(str==NULL)
	{
		msize=0;
		mstr=NULL;
	}
	else
	{
		msize=strlen(str);
		mstr=new char[msize+1];
		assert(mstr);
		strcpy(mstr,str);
	}
}

String::String(const String& str)
{
	if(str.msize==0)
	{
		msize=0;
		mstr=NULL;
	}
	else
	{
		msize=str.msize;
		mstr=new char[msize+1];
		assert(mstr);
		strcpy(mstr,str.mstr);
	}
}

String String::operator+(const String& str)
{
	String temp;
	temp.msize=msize+str.msize-1;
	temp.mstr=new char[temp.msize];
	assert(temp.mstr);
	strcpy(temp.mstr,mstr);
	strcpy(&(temp.mstr[msize]),str.mstr);
	return temp;
}

String& String::operator=(const String& str)
{
	if(this==&str)
		return *this;
	if(msize>0)
		delete []mstr;
	msize=str.msize;
	mstr=new char[msize];
	assert(mstr);
	strcpy(mstr,str.mstr);
	return *this;
}

void String::Display()
{
	cout<<mstr<<endl;
}

int main(int argc,char *argv[])
{
	char str1[]={"Hello"};
	char str2[]={"World"};
	String s1(str1),s2(str2),s3;
	s3=s1+s2;
	s3.Display();

	return 0;
}
如果用户没有定义一个类重载赋值运算符,编译器将生成一个缺省的赋值运算符。赋值运算符把源对象逐域地拷贝到目的对象;
下面做一个总结,编译器会自动生成的函数有:
1,构造函数;
2,析构函数;
3,拷贝构造函数;
4,赋值运算符;
拷贝构造函数和赋值运算符的区别是:
拷贝构造函数是要创建一个新的对象,而赋值运算符则是改变一个已经存在的对象的值;
需要注意的是当一个类包含指针类型的数据成员时,最好进行赋值运算符和拷贝构造函数的重载,否则很有可能造成指针悬挂问题(动态申请的空间无法访问,无法释放);
4,重载[]运算符
#include<iostream>
#include<cstring>
#include<cassert>

using namespace std;

class String
{
	public:
		String()
		{
			msize=0;
			mstr=NULL;
		}
		String(char *str);
		String(const String& str);
		String operator+(const String& str);
		String& operator=(const String& str);
		char& operator[](int index);//可读可写
		char operator[](int index)const;//只读
		void Display();
	private:
		char *mstr;
		int msize;
};

String::String(char *str)
{
	if(str==NULL)
	{
		msize=0;
		mstr=NULL;
	}
	else
	{
		msize=strlen(str);
		mstr=new char[msize+1];
		assert(mstr);
		strcpy(mstr,str);
	}
}

String::String(const String& str)
{
	if(str.msize==0)
	{
		msize=0;
		mstr=NULL;
	}
	else
	{
		msize=str.msize;
		mstr=new char[msize+1];
		assert(mstr);
		strcpy(mstr,str.mstr);
	}
}

String String::operator+(const String& str)
{
	String temp;
	temp.msize=msize+str.msize-1;
	temp.mstr=new char[temp.msize];
	assert(temp.mstr);
	strcpy(temp.mstr,mstr);
	strcpy(&(temp.mstr[msize]),str.mstr);
	return temp;
}

String& String::operator=(const String& str)
{
	if(this==&str)
		return *this;
	if(msize>0)
		delete []mstr;
	msize=str.msize;
	mstr=new char[msize];
	assert(mstr);
	strcpy(mstr,str.mstr);
	return *this;
}

char& String::operator[](int index)//可读可写
{
	return mstr[index-1];
}

char String::operator[](int index)const//只读,但只能由常对象调用
{
	return mstr[index-1];
}

void String::Display()
{
	cout<<mstr<<endl;
}

int main(int argc,char *argv[])
{
	char str1[]={"Hello"};
	char str2[]={"World"};
	String s1(str1),s2(str2),s3;
	s3=s1+s2;
	s3.Display();
	cout<<s3[2]<<endl;
	s3[2]=‘z‘;
	cout<<s3[2]<<endl;
	return 0;
}
重载[]运算符时,我写了两个版本,一个可读可写,另一个起只读作用;
注意:只能重载为类成员的操作符有四个=、[]、()、->
5,输入输出重载
#include<iostream>

using namespace std;

class Complex
{
	public:
		Complex()
		{
			re=0;
			im=0;
		}
		Complex(double r,double i)
		{
			re=r;
			im=i;
		}
		Complex operator+(const Complex& obj);
		Complex operator!();
		friend ostream& operator<<(ostream &os,const Complex &c);
		friend istream& operator>>(istream &is,Complex &c);
	private:
		double re;
		double im;
};

Complex Complex::operator+(const Complex& obj)
{
	Complex temp;
	temp.re=re+obj.re;
	temp.im=im+obj.im;
	return temp;
}

Complex Complex::operator!()
{
	Complex temp;
	temp.re=-re;
	temp.im=-im;
	return temp;
}

ostream& operator<<(ostream &os,const Complex &c)
{
	os<<c.re;
	if(c.im>0)
		os<<"+"<<c.im<<"i"<<endl;
	else
		os<<c.im<<"i"<<endl;
	return os;
}

istream& operator>>(istream &is,Complex &c)
{
	is>>c.re>>c.im;
	return is;
}

int main(int argc,char *argv[])
{
	Complex obj1(1,2),obj2(3,4);
	Complex obj3=obj1+!obj2;
	cout<<obj3;

	cin>>obj3;
	cout<<obj3;
	return 0;
}
输入输出运算符的重载只能用友元函数的方式,应为隐式调用为cout<<Class_obj其中Class_obj为类类型Class_name的对象;如果用显示调用,则变成了cout.operator<<Class_obj,很显然cout本身不是Class_name的对象;
为了保证输出运算符<<的连用性,重载函数的返回应该为ostream&;对于重载>>,需要注意的是第二个参数必须是对象的引用;

C++ Primer Plus学习笔记之运算符重载,布布扣,bubuko.com

C++ Primer Plus学习笔记之运算符重载

上一篇:【Javascript】OOP ‘终端' demo


下一篇:IOS 设置导航栏全局样式