operatpr
2016-06-16 17:46
405 查看
C++ operator关键字(重载操作符) operator是C++的关键字,它和运算符一起使用,表示一个运算符函数,理解时应将operator=整体上视为一个函数名。 这是C++扩展运算符功能的方法,虽然样子古怪,但也可以理解:一方面要使运算符的使用方法与其原来一致,另一方面扩展其功能只能通过函数的方式(c++中,“功能”都是由函数实现的)。 一、为什么使用操作符重载? 对于系统的所有操作符,一般情况下,只支持基本数据类型和标准库中提供的class,对于用户自己定义的class,如果想支持基本操作,比如比较大小,判断是否相等,等等,则需要用户自己来定义关于这个操作符的具体实现。比如,判断两个人是否一样大,我们默认的规则是按照其年龄来比较,所以,在设计person
这个class的时候,我们需要考虑操作符==,而且,根据刚才的分析,比较的依据应该是age。那么为什么叫重载呢?这是因为,在编译器实现的时候,已经为我们提供了这个操作符的基本数据类型实现版本,但是现在他的操作数变成了用户定义的数据类型class,所以,需要用户自己来提供该参数版本的实现。 二、如何声明一个重载的操作符? A: 操作符重载实现为类成员函数 重载的操作符在类体中被声明,声明方式如同普通成员函数一样,只不过他的名字包含关键字operator,以及紧跟其后的一个c++预定义的操作符。 可以用如下的方式来声明一个预定义的==操作符:
class person{ private: int age; public: person(int a){ this->age=a; }
inline bool operator == (const person &ps) const; };
实现方式如下:
inline bool person::operator==(const person &ps) const { if (this->age==ps.age) return true; return false; }
调用方式如下:
#include using namespace std; int main() { person p1(10); person p2(20); if(p1==p2) cout<<”the age is equal!”< return 0; }
这里,因为operator ==是class person的一个成员函数,所以对象p1,p2都可以调用该函数,上面的if语句中,相当于p1调用函数==,把p2作为该函数的一个参数传递给该函数,从而实现了两个对象的比较。 B:操作符重载实现为非类成员函数(全局函数) 对于全局重载操作符,代表左操作数的参数必须被显式指定。
例如: #include #include using namespace std; class person { public: int age; public: }; bool operator==(person const &p1 ,person const & p2) //满足要求,做操作数的类型被显示指定
{ if(p1.age==p2.age) return true; return false; } int main() { person rose; person jack; rose.age=18; jack.age=23; if(rose==jack) cout<<"ok"< return 0; }
C:如何决定把一个操作符重载为类成员函数还是全局名字空间的成员呢? ①如果一个重载操作符是类成员,那么只有当与他一起使用的左操作数是该类的对象时,该操作符才会被调用。如果该操作符的左操作数必须是其他的类型,则操作符必须被重载为全局名字空间的成员。 ②C++要求赋值=,下标[],调用(), 和成员指向-> 操作符必须被定义为类成员操作符。任何把这些操作符定义为名字空间成员的定义都会被标记为编译时刻错误。 ③如果有一个操作数是类类型如string类的情形那么对于对称操作符比如等于操作符最好定义为全局名字空间成员。
D:重载操作符具有以下限制: (
1) 只有C++预定义的操作符集中的操作符才可以被重载;
(
2
)对于内置类型的操作符,它的预定义不能被改变,应不能为内置类型重载操作符,如,
不能改变
int
型的操作符
+
的含义;
(
3
)
也不能为内置的数据类型定义其它的操作符;
(
4
)
只能重载类类型或枚举类型的操作符;
(
5
)
重载操作符不能改变它们的操作符优先级;
(
6
)
重载操作符不能改变操作数的个数;
(
7
)
除了对
( )
操作符外,对其他重载操作符提供缺省实参都是非法的;
E:
注意点
(
1
)后果载操操作符首先要确定它的返回值是左值,还是右值,如果是左值最返回引用,如
果是右值那就直接返回值;
(
2
)
+
号等这样的操作符没有对象可以容纳改变后值,对于这样的情况最好返回数值,否则
只能要操作符体内创建临时对象用于容纳改变后的值,如果在堆中创建临时对象返回指针或者
引用,在操作符函数体外还需要释放它,如果返回的对象而不是引用或者指针,那么效率是比
较低的。如果返回的是数值,最好在该类的构造函数中增加对该类型数值的转换函数,如:返
回值是
int
类型,那么最好有一个
int
类型作为参数的构造函数。
(
3
)在增量运算符中,放上一个整数形参,就是后增量运行符,它是值返回,对于前增量没
有形参,而且是引用返回,示例:
class Test
{
public:
Test(x=3){ m_value = x}
Test &operator ++(); //
前增量
Test &operator ++(int);//
后增量
private:
Int m_value:
};
Test &Test::operator ++()
{
m_value ++; //
先增量
return *this; //
返回当前对象
}
Test Test::operator ++(int)
{
Test tmp(*this); //
创建临时对象
m_value ++; //
再增量
这个class的时候,我们需要考虑操作符==,而且,根据刚才的分析,比较的依据应该是age。那么为什么叫重载呢?这是因为,在编译器实现的时候,已经为我们提供了这个操作符的基本数据类型实现版本,但是现在他的操作数变成了用户定义的数据类型class,所以,需要用户自己来提供该参数版本的实现。 二、如何声明一个重载的操作符? A: 操作符重载实现为类成员函数 重载的操作符在类体中被声明,声明方式如同普通成员函数一样,只不过他的名字包含关键字operator,以及紧跟其后的一个c++预定义的操作符。 可以用如下的方式来声明一个预定义的==操作符:
class person{ private: int age; public: person(int a){ this->age=a; }
inline bool operator == (const person &ps) const; };
实现方式如下:
inline bool person::operator==(const person &ps) const { if (this->age==ps.age) return true; return false; }
调用方式如下:
#include using namespace std; int main() { person p1(10); person p2(20); if(p1==p2) cout<<”the age is equal!”< return 0; }
这里,因为operator ==是class person的一个成员函数,所以对象p1,p2都可以调用该函数,上面的if语句中,相当于p1调用函数==,把p2作为该函数的一个参数传递给该函数,从而实现了两个对象的比较。 B:操作符重载实现为非类成员函数(全局函数) 对于全局重载操作符,代表左操作数的参数必须被显式指定。
例如: #include #include using namespace std; class person { public: int age; public: }; bool operator==(person const &p1 ,person const & p2) //满足要求,做操作数的类型被显示指定
{ if(p1.age==p2.age) return true; return false; } int main() { person rose; person jack; rose.age=18; jack.age=23; if(rose==jack) cout<<"ok"< return 0; }
C:如何决定把一个操作符重载为类成员函数还是全局名字空间的成员呢? ①如果一个重载操作符是类成员,那么只有当与他一起使用的左操作数是该类的对象时,该操作符才会被调用。如果该操作符的左操作数必须是其他的类型,则操作符必须被重载为全局名字空间的成员。 ②C++要求赋值=,下标[],调用(), 和成员指向-> 操作符必须被定义为类成员操作符。任何把这些操作符定义为名字空间成员的定义都会被标记为编译时刻错误。 ③如果有一个操作数是类类型如string类的情形那么对于对称操作符比如等于操作符最好定义为全局名字空间成员。
D:重载操作符具有以下限制: (
1) 只有C++预定义的操作符集中的操作符才可以被重载;
(
2
)对于内置类型的操作符,它的预定义不能被改变,应不能为内置类型重载操作符,如,
不能改变
int
型的操作符
+
的含义;
(
3
)
也不能为内置的数据类型定义其它的操作符;
(
4
)
只能重载类类型或枚举类型的操作符;
(
5
)
重载操作符不能改变它们的操作符优先级;
(
6
)
重载操作符不能改变操作数的个数;
(
7
)
除了对
( )
操作符外,对其他重载操作符提供缺省实参都是非法的;
E:
注意点
(
1
)后果载操操作符首先要确定它的返回值是左值,还是右值,如果是左值最返回引用,如
果是右值那就直接返回值;
(
2
)
+
号等这样的操作符没有对象可以容纳改变后值,对于这样的情况最好返回数值,否则
只能要操作符体内创建临时对象用于容纳改变后的值,如果在堆中创建临时对象返回指针或者
引用,在操作符函数体外还需要释放它,如果返回的对象而不是引用或者指针,那么效率是比
较低的。如果返回的是数值,最好在该类的构造函数中增加对该类型数值的转换函数,如:返
回值是
int
类型,那么最好有一个
int
类型作为参数的构造函数。
(
3
)在增量运算符中,放上一个整数形参,就是后增量运行符,它是值返回,对于前增量没
有形参,而且是引用返回,示例:
class Test
{
public:
Test(x=3){ m_value = x}
Test &operator ++(); //
前增量
Test &operator ++(int);//
后增量
private:
Int m_value:
};
Test &Test::operator ++()
{
m_value ++; //
先增量
return *this; //
返回当前对象
}
Test Test::operator ++(int)
{
Test tmp(*this); //
创建临时对象
m_value ++; //
再增量
相关文章推荐
- Linux服务器上安装织梦CMS
- 从读取视频程序比较opencv1.0与2.0以上版本
- WordPress小工具开发教程(网站公告)
- Linux中查看jdk版本
- 第三部分 linux busybox最小文件系统的移植
- Zabbix 3.0 监控交换机(4)——温度、风扇监控及配置Trigger
- 【Linux】ACL权限管理
- 【VMCloud云平台】拥抱Docker(五)Volume(2)
- linux sed 批量替换字符串
- Windows玩转Docker
- OpenCV遍历文件夹下所有文件
- 修改linux终端命令行颜色
- 进度条---shell脚本版本
- Linux mysql 基础操作
- tomcat发布网站的三种方式
- CentOS7之搭建lnmp环境
- nginx平滑重启
- tomcat的安装目录结构简介
- 免注册,下载NI网站的驱动程序包
- j2ee工程(即动态网站工程Dynamic Web Project),在tomcat容器下运行时(确切说是class文件在运行时),jar包的执行顺序