0、重载一元操作符的方法

一元运算符即只需要一个操作用的运算符,如取地址运算符(&)、复数(-)、自减(--)、自加(++)等。

运算符重载可以分为3种方式:类的非静态成员函数、类的友元函数、普通函数。

例如有 1 个操作数 a,一元运算符 ? (表示一个一元运算符),a? 或者?a 的操作会被解释为下面2种形式之一

//a? or ?a
a.operator?(); //类的非静态成员函数
operator?(a); //友元函数 和 普通函数

第一种形式是运算符被重载为类的非静态成员函数,没有参数。

这种方式要求操作数(即a)必须是一个对象,operator?是这个对象的非静态成员函数,

第二种形式是运算符被重载为类的友元函数 或 普通函数,有1个参数。

重载为 类的友元函数 和 普通函数的区别是 类的友元函数可以直接访问类的私有成员,而普通函数不可以。

1、重载++、--(前置 和 后置)

自增运算符(++)和 自减运算符(--)都有 前置 和 后置 2中情况。

C++中为了区分前置和后置规定:前置重载与普通运算符重载一致,后置重载需要在参数列表中加入一个无用的参数。

++/-- 的3种重载方式如下: 前置的情况可以返回类的引用,后置的情况只能返回临时对象。

//类的非静态成员函数
class X
{
X& operator++(){...} //前置++
X operator++(int){...} //后置++ X& operator--(){...}   //前置--  
X operator--(int){...} //后置--
}
/*-----------------------------------------------------*/
//类的友元函数
class X
{
friend X& operator++(X& ref){...} //前置++
friend X operator++(X& ref, int){...} //后置++ friend X& operator--(X& ref){...} //前置--
friend X operator--(X& ref, int){...} //后置--
}
/*-----------------------------------------------------*/
//普通函数
class X
{
...
}
X& operator++(X& ref){...} //前置++
X operator++(X& ref, int){...} //后置++ X& operator--(X& ref){...} //前置--
X operator--(X& ref, int){...} //后置--

下例中有3个Counter类 CounterA、CounterB 和CounterC,3个类都重载了前置++、-- 和后置++、--。

其中CounterA使用类的非静态成员函数方式重载,CounterB使用类的友元函数方式重载,CounterC使用普通函数方式重载。

#include <iostream>
using namespace std; class CounterA
{
public:
//默认构造函数(Default constructor)
CounterA(){cout<<"Default Constructor"<<endl;}
//带参数的构造函数(The constructor with parameters)
CounterA(int m):n(m){cout<<"Parameter Constructor"<<endl;}
//拷贝构造函数(Copy constructor)
CounterA(const CounterA& ref){n = ref.n; cout<<"Copy Constructor"<<endl;}
//析构函数(destructor)
~CounterA(){cout<<"Destructor"<<endl;} //重载运算符 ++(前置)
const CounterA& operator++(){++n; return *this;}
//重载运算符 ++(后置)
const CounterA operator++(int dump){CounterA Tmp(*this);++n; return Tmp;}
//重载运算符 --(前置)
const CounterA& operator--(){--n; return *this;}
//重载运算符 --(后置)
const CounterA operator--(int dump){CounterA Tmp(*this);--n; return Tmp;} //输出用函数
void display(void){cout<<"n = " << n << endl;}
//GetData
int GetVal(void){return n;} private:
int n;
}; class CounterB
{
public:
//默认构造函数(Default constructor)
CounterB(){cout<<"Default Constructor"<<endl;}
//带参数的构造函数(The constructor with parameters)
CounterB(int m):n(m){cout<<"Parameter Constructor"<<endl;}
//拷贝构造函数(Copy constructor)
CounterB(const CounterB& ref){n = ref.n; cout<<"Copy Constructor"<<endl;}
//析构函数(destructor)
~CounterB(){cout<<"Destructor"<<endl;} //重载运算符 ++(前置)
friend const CounterB& operator++(CounterB& ref){ref.n += ;return ref;}
//重载运算符 ++(后置)
friend const CounterB operator++(CounterB& ref, int dump){CounterB Tmp(ref);ref.n += ; return Tmp;}
//重载运算符 --(前置)
friend const CounterB& operator--(CounterB& ref){ref.n -= ;return ref;}
//重载运算符 --(后置)
friend const CounterB operator--(CounterB& ref, int dump){CounterB Tmp(ref);ref.n -= ; return Tmp;} //输出用函数
void display(void){cout<<"n = " << n << endl;}
//GetData
int GetVal(void){return n;}
//SetData
void SetVal(int val){n = val;} private:
int n;
}; class CounterC
{
public:
//默认构造函数(Default constructor)
CounterC(){cout<<"Default Constructor"<<endl;}
//带参数的构造函数(The constructor with parameters)
CounterC(int m):n(m){cout<<"Parameter Constructor"<<endl;}
//拷贝构造函数(Copy constructor)
CounterC(const CounterC& ref){n = ref.n; cout<<"Copy Constructor"<<endl;}
//析构函数(destructor)
~CounterC(){cout<<"Destructor"<<endl;} //输出用函数
void display(void){cout<<"n = " << n << endl;}
//GetData
int GetVal(void){return n;}
//SetData
void SetVal(int val){n = val;} private:
int n;
};
//重载运算符 ++(前置)
const CounterC& operator++(CounterC& ref){ref.SetVal(ref.GetVal()+);return ref;}
//重载运算符 ++(后置)
const CounterC operator++(CounterC& ref, int dump){CounterC Tmp(ref);ref.SetVal(ref.GetVal()+); return Tmp;}
//重载运算符 --(前置)
const CounterC& operator--(CounterC& ref){ref.SetVal(ref.GetVal()-);return ref;}
//重载运算符 --(后置)
const CounterC operator--(CounterC& ref, int dump){CounterC Tmp(ref);ref.SetVal(ref.GetVal()-); return Tmp;} int main(void)
{
CounterA cp1();cp1.display();
++cp1;cp1.display();
--cp1;cp1.display();
CounterA cp2(cp1++);cp2.display();
CounterA cp3(cp1--);cp3.display(); CounterB cp4();cp4.display();
++cp4;cp4.display();
--cp4;cp4.display();
CounterB cp5(cp4++);cp5.display();
CounterB cp6(cp4--);cp6.display(); CounterC cp7();cp7.display();
++cp7;cp7.display();
--cp7;cp7.display();
CounterC cp8(cp7++);cp8.display();
CounterC cp9(cp7--);cp9.display(); return ;
}

2、重载负号(-)

下例中有3个Complex类 ComplexA、ComplexB 和ComplexC,3个类都重载了负号。

其中ComplexA使用类的非静态成员函数方式重载,ComplexB使用类的友元函数方式重载,ComplexC使用普通函数方式重载。

#include <iostream>
using namespace std; class ComplexA
{
public:
//constructor
ComplexA(double re, double im):real(re),image(im){}
//Operator Overload : -
ComplexA operator-(){return ComplexA(-real, -image);}
//display
void display(void){if (image>)cout<<real<<"+"<<image<<endl;else cout<<real<<image<<endl;} private:
double real;
double image;
}; class ComplexB
{
public:
//constructor
ComplexB(double re, double im):real(re),image(im){}
//Operator Overload : -
friend ComplexB operator-(ComplexB& ref){return ComplexB(-ref.real, -ref.image);}
//display
void display(void){if (image>)cout<<real<<"+"<<image<<endl;else cout<<real<<image<<endl;}
private:
double real;
double image;
}; class ComplexC
{
public:
//constructor
ComplexC(double re, double im):real(re),image(im){}
//GetData
double GetReal(void){return real;}
double GetImage(void){return image;}
//display
void display(void){if (image>)cout<<real<<"+"<<image<<endl;else cout<<real<<image<<endl;} private:
double real;
double image;
}; //Operator Overload : -
ComplexC operator-(ComplexC& ref){return ComplexC(-ref.GetReal(), -ref.GetImage());} int main(void)
{
ComplexA a1(,);a1.display();
ComplexA a2(-a1);a2.display(); ComplexB b1(,);b1.display();
ComplexB b2(-b1);b2.display(); ComplexC c1(,);c1.display();
ComplexC c2(-c1);c2.display(); return ;
}

C++运算符重载——重载一元运算符的更多相关文章

  1. C++ 重载运算符和重载函数

    C++ 重载运算符和重载函数 C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载. 重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是 ...

  2. C++解析七-重载运算符和重载函数

    重载运算符和重载函数C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载.重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它们的参数列 ...

  3. C++ 自增、自减运算符的重载和性能分析

    01 ++.--运算符重载函数的格式 自增运算符和自减运算符是有前置和后置之分的,如: a++ // 后置自增运算符 ++a // 前置自增运算符 b-- // 后置自减运算符 --b // 前置自减 ...

  4. C++基础之自增和自减运算符的重载

    1. 格式 1.1 分为前置和后置格式: int x = 0; int y = 0; // 后置自增运算符 x++; // 前置自增运算符 ++x; // 后置自减运算符 y--; // 前置自减运算 ...

  5. C++运算符重载——重载二元运算符

    1.重载二元操作符的方法 二元运算符又称为双目运算符,即需要2个操作数的运算符,例如 + - * / 等. 运算符重载可以分为3种方式:类的非静态成员函数.类的友元函数.普通函数. 例如有 2 个操作 ...

  6. [置顶] C++基础之六:运算符的重载

    网上太多有关运算符的重载了,但是写的太过的详细,不适合新手入门,特别是那什么++和--的前增量后增量重载,一元二元运算符重载,特殊运算符,下标运算符,new和delete,甚至是指针运算符的重载,吓退 ...

  7. 【转】C++对成员访问运算符->的重载

    运算符->的重载比较特别,它只能是非静态的成员函数形式,而且没有参数. 1.如果返回值是一个原始指针,那么就将运算符的右操作数当作这个原始指针所指向类型的成员进行访问: 2.如果返回值是另一个类 ...

  8. C++题目一道: 重载`->': 您真的懂成员访问运算符的重载吗?

    原题目在这里: http://hi.baidu.com/shilyx/item/672736e14a14a90c64db003a 要求: //给出类Test的定义和实现,使程序编译通过, //并且ma ...

  9. c++的运算符的重载

    1 什么是c++运算符的重载 c++运算符的重载就是说对+.-.>.<等运算符进行重新定义,这样的话,除了基本的类型,所有的类都可以进行基本的运算了,用起来非常方便.特别是用在各种算法中. ...

随机推荐

  1. Quartz.NET Windows

    Quartz.NET Windows 服务示例 想必大家在项目中处理简单的后台持续任务或者定时触发任务的时候均使用 Thread 或者 Task 来完成,但是项目中的这种需求一旦多了的话就得将任务调度 ...

  2. java中的substring用法

    String str="我是中国人"; str = str.substring(0, 2) +"_"+str.substring(3, 4); 结果:str=& ...

  3. 使用sqlserver日期函数获取当前日期

    使用sqlserver日期函数中的getdate()可以获取当现的日期,下面就将为您介绍这种使用sqlserver日期函数获取当前日期的方法,供您参考,希望对您学习sqlserver日期函数有所启迪. ...

  4. .NET Framework 4.5、4.5.1 和 4.5.2 中的新增功能

    .NET Framework 4.5.4.5.1 和 4.5.2 中的新增功能 https://msdn.microsoft.com/zh-cn/library/ms171868.aspx

  5. bzoj 1565 最大权闭合子图

    因为每个植物都有保护的点(每排相邻的右面的算是保护左面的),所以连他和保护 的点一条边,然后每个点有自己的权值,要找到最大的权值且满足每个点在访问时他 的前驱一定被访问,那么反向建边,转化为后继必须访 ...

  6. tomcat集群 (自带Cluster集群)

    不用借助其他任何工具,tomcat自身就可以实现session共享,实现集群.以下为大概步骤 1,如果是在同一台机器上,请保持多个tomcat端口(一个tomcat对应三个端口)不相同:如果是不同机器 ...

  7. 数位DP之小小结

    资料链接:http://wenku.baidu.com/view/9de41d51168884868662d623.html http://wenku.baidu.com/view/d2414ffe0 ...

  8. 为现代JavaScript开发做好准备

    今天无论是在浏览器中还是在浏览器外,JavaScript世界正在经历翻天覆地地变化.如果我们谈论脚本加载.客户端的MVC框架.压缩器.AMD.Common.js还有Coffeescript……只会让你 ...

  9. javascript实现数据结构与算法系列:功能完整的线性链表

    由于链表在空间的合理利用上和插入,删除时不需要移动等的有点,因此在很多场合下,它是线性表的首选存储结构.然而,它也存在着实现某些基本操作,如求线性表长度时不如顺序存储结构的缺点:另一方面,由于在链表中 ...

  10. LCT小结

    LCT真是灵活好用… LCT的基本思想与树链剖分差不多,都是把树剖成一条条链,只不过LCT用的是SPLAY维护的,而且,SPLAY的链是会变化的,不像剖分是定死的. LCT最重要的操作就是access ...