返回值类型 operator 运算符(形参表){
……
}
//示例
class Complex{
public:
double real,imag;
Complex( double r = 0.0, double i= 0.0 ):real(r),imag(i) { }
Complex operator-(const Complex & c);
};
Complex operator+( const Complex & a, const Complex & b){
return Complex( a.real+b.real,a.imag+b.imag); // 返回一个临时对象
}
Complex Complex::operator-(const Complex & c){
return Complex(real - c.real, imag - c.imag); // 返回一个临时对象
}
int main(){
Complex a(4,4),b(1,1),c;
c = a + b; // 等价于c=operator+(a,b);
cout << c.real << "," << c.imag << endl;
cout << (a-b).real << "," << (a-b).imag << endl;
//a-b 等价于a.operator-(b)
return 0;
}
class String {
private:
char * str;
public:
String():str(new char[1]) { str[0] = 0;}
const char * c_str() { return str; };
String & operator=(const char * s);
String::~String() { delete [] str; }
};
String & String::operator = (const char * s){
// 重载“=”得 以使得 obj = “hello” 能够成立
delete [] str;
str = new char[strlen(s)+1];
strcpy( str, s);
return * this;
}
int main(){
String s;
s = "Good Luck," ; //等价于 s.operator=("Good Luck,");
cout << s.c_str() << endl;
// String s2 = "hello!"; // 这条语句要是不注释掉就会出错
s = "Shenzhou 8!"; //等价于 s.operator=("Shenzhou 8!");
cout << s.c_str() << endl;
return 0;
}
上述代码实现了两个功能:将 char* 类型赋给string对象;返回值为String&类型,故可以作为左值。但还需要解决以下问题:
因此需要加上两个成员函数:赋值运算符的重载函数和复制构造函数。
//赋值运算符的重载函数
String & operator = (const String & s){
if( this == & s) return * this; //排除自身赋给自身的情况
delete [] str;
str = new char[strlen(s.str)+1];
strcpy( str,s.str);
return * this;
}
//复制构造函数
String( String & s){
str = new char[strlen(s.str)+1];
strcpy(str,s.str);
}
另外:
//要编写可变长整型数组类,使之能如下使用:
int main() {
CArray a; //开始里的数组是空的
for( int i = 0;i < 5;++i)
a.push_back(i);
CArray a2,a3;
a2 = a;
for( int i = 0; i < a.length(); ++i )
cout << a2[i] << " " ;
a2 = a3; //a2是空的
for( int i = 0; i < a2.length(); ++i ) //a2.length()返回0
cout << a2[i] << " ";
cout << endl;
a[3] = 100;
CArray a4(a);
for( int i = 0; i < a4.length(); ++i )
cout << a4[i] << " ";
return 0;
}
//程序输出结果为
//0 1 2 3 4
//0 1 2 100 4
下面的代码可以满足当前要求:
#include <iostream>
#include <cstring>
using namespace std;
class CArray{
int* p;
int n;
public:
CArray(int s=0);
CArray(CArray& a);
~CArray(){ if(n>0) delete []p;}
CArray& operator=(const CArray& a);
int& operator[](int i){if(i>=0 && i<n) return p[i];}
void push_back(int k);
int length(){return n;}
};
CArray::CArray(int s):n(s){
if(n == 0) p = NULL;
else p = new int[n];
}
CArray::CArray(CArray& a){
n = a.n;
if(n == 0){
p = NULL;
}else{ //n>0
p = new int[n];
memcpy(p, a.p, sizeof(int)*n);
}
}
CArray& CArray::operator=(const CArray& a){
if(this == &a) return *this;
//若原空间不足,则需要重新开辟空间,否则不需要
if(n < a.n){ //n可能为0
if(n > 0) delete []p;
p = new int[a.n];
}
//复制数据
if(a.n == 0){
p = NULL;
}else{ //a.n>0
memcpy(p, a.p, sizeof(int)*a.n);
}
n = a.n;
return *this;
}
void CArray::push_back(int k){
//if(n > 0) delete []p;
int* temp = new int[n+1];
memcpy(temp, p, sizeof(int)*n);
temp[n] = k;
if(n > 0) delete []p;
p = temp;
n++;
}
int main() {
CArray a; //开始里的数组是空的
for( int i = 0;i < 5;++i)
a.push_back(i);
CArray a2,a3;
a2 = a;
for( int i = 0; i < a.length(); ++i )
cout << a2[i] << " " ;
a2 = a3; //a2是空的
for( int i = 0; i < a2.length(); ++i ) //a2.length()返回0
cout << a2[i] << " ";
cout << endl;
a[3] = 100;
CArray a4(a);
for( int i = 0; i < a4.length(); ++i )
cout << a4[i] << " ";
return 0;
}
cout 是ostream 类的对象,是在 iostream 中定义的。
“<<” 能用在cout上是因为,在iostream里对 “<<” 进行了重载。
cout << 5 << “this” 的本质就是cout.operator<<(5).operator<<(“this”);
//重载成成员函数
ostream & ostream::operator<<(int n){
…… // 输出n的代码
return * this;
}
//重载成全局函数
ostream & operator<<( ostream & o, const CStudent & s){
o << s.nAge ;
return o;
}
类型强制转换运算符被重载时不能写返回值类型,实际上其返回值类型就是该类型强制转换运算符代表的类型。
#include <iostream>
using namespace std;
class Complex{
double real,imag;
public:
Complex(double r=0,double i=0):real(r),imag(i) { };
operator double () { return real; } //重载强制类型转换运算符 double
};
int main(){
Complex c(1.2,3.4);
cout << (double)c << endl; //输出 1.2
double n = 2 + c; //等价于 double n=2+c.operator double()
cout << n; //输出 3.2
}
自增运算符++、自减运算符--有前置/后置之分,为了区分所重载的是前置运算符还是后置运算符,C++规定:
此处需注意:
class CDemo {
private :
int n;
public:
CDemo(int i=0):n(i) { }
CDemo & operator++(); // 用于前置形式
CDemo operator++( int ); // 用于后置形式
operator int ( ) { return n; } //强制类型转换运算符的重载
friend CDemo & operator--(CDemo & );
friend CDemo operator--(CDemo & ,int);
};
CDemo & CDemo::operator++(){ //前置 ++
n ++;
return * this;
} // ++s即为: s.operator++();
CDemo CDemo::operator++( int k ){ //后置 ++
CDemo tmp(*this); // 记录修改前的对象
n ++;
return tmp; // 返回修改前的对象
} // s++即为: s.operator++(0);
CDemo & operator--(CDemo & d){// 前置--
d.n--;
return d;
} //--s即为: operator--(s);
CDemo operator--(CDemo & d,int){// 后置--
CDemo tmp(d);
d.n --;
return tmp;
} //s--即为: operator--(s, 0);
int main(){
CDemo d(5);
cout << (d++ ) << ","; //于 等价于 d.operator++(0);
cout << d << ",";
cout << (++d) << ","; //于 等价于 d.operator++();
cout << d << endl;
cout << (d-- ) << ","; //于 等价于 operator--(d,0);
cout << d << ",";
cout << (--d) << ","; //于 等价于 operator--(d);
cout << d << endl;
return 0;
}
//输出结果:
//5,6,7,7
//7,6,5,5
对于运算符的重载,注意事项:
需要掌握String对象和可变长数组对象的设计。
//001:MyString http://cxsjsxmooc.openjudge.cn/2019t3fall4/001/
#include <iostream>
#include <cstring>
using namespace std;
class MyString {
char * p;
public:
MyString(const char * s) {
if(s){
p = new char[strlen(s) + 1];
strcpy(p,s);
}else
p = NULL;
}
~MyString() { if(p) delete [] p; }
//your code start here
MyString(const MyString& str1){
p = new char[strlen(str1.p)+1];
strcpy(p, str1.p);
}
MyString& operator=(MyString& str1){
if(this==&str1) return *this;
if(p) delete[]p;
p = new char[strlen(str1.p)+1]; //即便str1.p为空也可以
strcpy(p, str1.p);
return *this;
}
MyString& operator=(const char* s){
if(p) delete []p;
p = new char[strlen(s)+1];
strcpy(p, s);
return *this;
}
void Copy(const char* s){
if(p) delete []p;
p = new char[strlen(s)+1];
strcpy(p, s);
}
friend ostream& operator<<(ostream& o, MyString& str1){
cout << str1.p;
return o;
}
//your code end here
};
int main(){
char w1[200],w2[100];
cin >> w1 >> w2;
MyString s1(w1),s2 = s1;
MyString s3(NULL);
s3.Copy(w1);
cout << s1 << "," << s2 << "," << s3 << endl;
s2 = w2;
s3 = s2;
s1 = s3;
cout << s1 << "," << s2 << "," << s3 << endl;
}
原文:https://www.cnblogs.com/inchbyinch/p/12233222.html