C++ 重载运算符和重载函数
C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载。
重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它们的参数列表和定义(实现)不相同。
当您调用一个重载函数或重载运算符时,编译器通过把您所使用的参数类型与定义中的参数类型进行比较,决定选用最合适的定义。选择最合适的重载函数或重载运算符的过程,称为重载决策。
C++ 中的函数重载
在同一个作用域内,可以声明几个功能类似的同名函数,但是这些同名函数的形式参数(指参数的个数、类型或者顺序)必须不同。您不能仅通过返回类型的不同来重载函数。
下面的实例中,同名函数 print() 被用于输出不同的数据类型:
实例
当上面的代码被编译和执行时,它会产生下列结果:
C++ 中的运算符重载
您可以重定义或重载大部分 C++ 内置的运算符。这样,您就能使用自定义类型的运算符。
重载的运算符是带有特殊名称的函数,函数名是由关键字 operator 和其后要重载的运算符符号构成的。与其他函数一样,重载运算符有一个返回类型和一个参数列表。
声明加法运算符用于把两个 Box 对象相加,返回最终的 Box 对象。大多数的重载运算符可被定义为普通的非成员函数或者被定义为类成员函数。如果我们定义上面的函数为类的非成员函数,那么我们需要为每次操作传递两个参数,如下所示:
下面的实例使用成员函数演示了运算符重载的概念。在这里,对象作为参数进行传递,对象的属性使用 this 运算符进行访问,如下所示:
实例
当上面的代码被编译和执行时,它会产生下列结果:
可重载运算符/不可重载运算符
下面是可重载的运算符列表:
|双目算术运算符 |+ (加),-(减),(乘),/(除),% (取模)|
|—|—|
|关系运算符 |==(等于),!= (不等于),< (小于),> (大于>,<=(小于等于),>=(大于等于)|
|逻辑运算符 |||(逻辑或),&&(逻辑与),!(逻辑非)|
|单目运算符 |+ (正),-(负),(指针),&(取地址)|
|自增自减运算符 |++(自增),–(自减)|
|位运算符 | | (按位或),& (按位与),~(按位取反),^(按位异或),,<< (左移),>>(右移)|
|赋值运算符| =, +=, -=, =, /= , % = , &=, |=, ^=, <<=, >>=|
|空间申请与释放 |new, delete, new[ ] , delete[]|
|其他运算符 | ()(函数调用),->(成员访问),->(成员指针访问),,(逗号),[](下标)|
下面是不可重载的运算符列表:
运算符重载实例
下面提供了各种运算符重载的实例,帮助您更好地理解重载的概念。
序号 运算符和实例
1 一元运算符重载
2 二元运算符重载
3 关系运算符重载
4 输入/输出运算符重载
5 ++ 和 – 运算符重载
6 赋值运算符重载
7 函数调用运算符 () 重载
8 下标运算符 [] 重载
9 类成员访问运算符 -> 重载
1.C++ 一元运算符重载
一元运算符只对一个操作数进行操作,下面是一元运算符的实例:
一元运算符通常出现在它们所操作的对象的左边,比如 !obj
、-obj
和++obj
,但有时它们也可以作为后缀,比如 obj++
或obj--
。
下面的实例演示了如何重载一元减运算符( - )。
实例
当上面的代码被编译和执行时,它会产生下列结果:
2.C++ 二元运算符重载
二元运算符需要两个参数,下面是二元运算符的实例。我们平常使用的加运算符( + )、减运算符( - )、乘运算符( * )和除运算符( / )都属于二元运算符。就像加(+)运算符。
下面的实例演示了如何重载加运算符( + )。类似地,您也可以尝试重载减运算符( - )和除运算符( / )。
当上面的代码被编译和执行时,它会产生下列结果:
3.C++ 关系运算符重载
C++ 语言支持各种关系运算符( < 、 > 、 <= 、 >= 、 == 等等)
,它们可用于比较 C++ 内置的数据类型。
您可以重载任何一个关系运算符,重载后的关系运算符可用于比较类的对象。
下面的实例演示了如何重载 < 运算符,类似地,您也可以尝试重载其他的关系运算符。
当上面的代码被编译和执行时,它会产生下列结果:
4.C++ 输入/输出运算符重载
C++ 能够使用流提取运算符 >>
和流插入运算符 <<
来输入和输出内置的数据类型。您可以重载流提取运算符和流插入运算符来操作对象等用户自定义的数据类型。
在这里,有一点很重要,我们需要把运算符重载函数声明为类的友元函数,这样我们就能不用创建对象而直接调用函数。
下面的实例演示了如何重载提取运算符 >>
和插入运算符 <<
。
当上面的代码被编译和执行时,它会产生下列结果:
5.C++ ++ 和 – 运算符重载
递增运算符( ++ )和递减运算符( – )是 C++ 语言中两个重要的一元运算符。
下面的实例演示了如何重载递增运算符( ++ ),包括前缀和后缀两种用法。类似地,您也可以尝试重载递减运算符( - - )。
实例
当上面的代码被编译和执行时,它会产生下列结果:
6.C++ 赋值运算符重载
就像其他运算符一样,您可以重载赋值运算符( = ),用于创建一个对象,比如拷贝构造函数。
下面的实例演示了如何重载赋值运算符。
当上面的代码被编译和执行时,它会产生下列结果:
7.C++ 函数调用运算符 () 重载
函数调用运算符 () 可以被重载用于类的对象。当重载 () 时,您不是创造了一种新的调用函数的方式,相反地,这是创建一个可以传递任意数目参数的运算符函数。
下面的实例演示了如何重载函数调用运算符 ()。
当上面的代码被编译和执行时,它会产生下列结果:
8.C++ 下标运算符 [] 重载
下标操作符 [] 通常用于访问数组元素。重载该运算符用于增强操作 C++ 数组的功能。
下面的实例演示了如何重载下标运算符 []。
当上面的代码被编译和执行时,它会产生下列结果:
9.C++ 类成员访问运算符 -> 重载
类成员访问运算符( -> )可以被重载,但它较为麻烦。它被定义用于为一个类赋予”指针”行为。运算符 -> 必须是一个成员函数。如果使用了 -> 运算符,返回类型必须是指针或者是类的对象。
运算符 -> 通常与指针引用运算符 * 结合使用,用于实现”智能指针”的功能。这些指针是行为与正常指针相似的对象,唯一不同的是,当您通过指针访问对象时,它们会执行其他的任务。比如,当指针销毁时,或者当指针指向另一个对象时,会自动删除对象。
间接引用运算符 -> 可被定义为一个一元后缀运算符。也就是说,给出一个类:
语句p->m
被解释为 (p.operator->())->m
。同样地,下面的实例演示了如何重载类成员访问运算符 ->
。
当上面的代码被编译和执行时,它会产生下列结果: