操作符重载(operator overloading)是一种形式的C++多态,C++将操作符重载扩展到用户自定义的类型,如允许使用+将两个自定义的对象相加,编译器将根据操作数的数目和类型决定使用那种加法定义。

要重载操作符,需使用操作符函数,格式如下:

operator op (argument-list)

op:要重载的操作符

argument-list: 操作数

操作符函数可以是类的成员函数,也可以是友元函数,如果是类成员函数,则第一个操作数是调用对象,它不在argument-list中。

操作符重载范例,定义一个时间类Time,重载 + - *操作符。

//Time.h

#ifndef TIME_H_
#define TIME_H_
#include <iostream>
class Time
{
public:
Time(void);
Time(int h,int m = );
~Time(void);
void AddMin(int m);
void AddHr(int h);
void Reset(int h = , int m = );
void Show()const;
//重载 +操作符
Time operator+ (const Time & t)const;
Time operator- (const Time & t)const;
// *
Time operator* (double n) const;
//友元函数
friend Time operator* (double n, const Time & t);
//重载<<操作符
friend std::ostream & operator<< (std::ostream & os, const Time & t); private:
int m_hours;
int m_minutes;
};
#endif

//具体实现 Time.cpp

 #include "Time.h"

 Time::Time(void)
:m_hours()
,m_minutes()
{
} Time::Time(int h,int m)
:m_hours(h)
,m_minutes(m)
{
} Time::~Time(void)
{
} void Time:: AddMin(int m)
{
m_minutes += m;
m_hours += m_minutes / ;
m_minutes = m_minutes % ;
}
void Time::AddHr(int h)
{
m_hours += h;
}
void Time::Reset(int h, int m)
{
m_hours = h;
m_minutes = m;
}
void Time::Show()const
{
std::cout << m_hours << ":" << m_minutes;
}
Time Time::operator+ (const Time & t)const
{
Time sum;
sum.m_minutes = m_minutes + t.m_minutes;
sum.m_hours = m_hours + t.m_hours + sum.m_minutes / ;
sum.m_minutes = sum.m_minutes % ;
return sum;
}
Time Time::operator- (const Time & t)const
{
Time diff;
int total1,total2;
total1 = m_minutes + m_hours * ;
total2 = t.m_minutes + t.m_hours * ;
diff.m_hours = (total1 - total2) / ;
diff.m_minutes = (total1 - total2) % ;
return diff;
}
Time Time::operator* (double n) const
{
Time result;
long total_minutes = m_hours * * n + m_minutes * n;
result.m_hours = total_minutes / ;
result.m_minutes = total_minutes % ;
return result;
} Time operator* (double n, const Time & t)
{
Time result;
long total_minutes = t.m_hours * * n + t.m_minutes * n;
result.m_hours = total_minutes / ;
result.m_minutes = total_minutes % ;
return result;
} std::ostream & operator<< (std::ostream & os, const Time & t)
{
os << t.m_hours << ":" << t.m_minutes;
return os;
}
Time operator- (const Time & t)const;
重载减法操作符后,可以很方便对Time对象进行减法运算
Time start_time(10,40);
Time end_time(11,50); Time diff_time = end_time - start_time;
操作符函数为成员函数时,操作符左侧的对象是调用对象,右侧的为参数传递的对象,如上述减法操作实际上等同于:
end_time.operator-(start_time); 关于操作符重载:
1、重载限制
1)重载后的操作符必须至少有一个操作数是自定义类型,主要是防止用户为标准类型重载操作符。
2)使用操作符不能违反操作符原来的句法规则。如 % 是二元操作符,需要两个操作数,不能将它重载成使用一个操作数。
3)不能修改操作符的优先级
4)不能定义新的操作符
5)以下操作符不能重载:
sizeof
.
.*        --成员指针操作符
::
?:         --条件运算符
typeid       --一个RTTI操作符
const_cast    --强制类型转换符
dynamic_cast
reinterpret_cast
static_cast
6)大部分操作符都可以通过成员函数或非成员函数进行重载,但下面的操作符只能通过成员函数重载:
= 赋值操作符
() 函数调用操作符
[] 下标操作符
-> 通过指针访问类成员的操作符 2、通过友元函数重载
对于很多操作符来说,可以选择使用成员函数或非成员函数来实现操作符重载。一般来说非成员函数应该是友元函数,这样才可能直接访问类的私有数据。
如对于Time类的加法操作符: 使用类成员函数重载
Time operator+ (const Time & t)const; 使用非成员函数重载:
friend Time operator+ (const Time & t1, const Time & t2); 对于成员函数版本,一个操作数通过this指针隐式传递,一个通过参数表
友元版本,两个操作数都作为参数来传递
这两种重载方式都能匹配 T1+T2
但在重载操作符中,只能选择其中的一种格式,同时定义被视为二义性错误,导致编译错误。 有些情况必须使用友元来重载
在Time类中,*乘法操作符与其他两种重载的操作符不同,它使用了两种不同的类型,将一个Time值与一个double值结合在一起,这限制了该操作符的使用方式
*使用成员函数重载如下:
Time operator* (double n) const;
意味着只能这样使用该操作符 A = B * 2.5 --> A = B.operator*(2.5) (注:左侧操作数应是调用对象) 如果这样使用将导致编译错误 A = 2.5 * B (左侧操作数是调用对象,而2.5不是对象) 这种情况可以用非成员函数来解决,由于要访问私有数据,故定义为友元函数:
friend Time operator* (double n, const Time & t); 此时,A = 2.5 * B --> A = operator*(2.5,B)
对于非成员重载操作符函数,op左边的操作数对应第一个参数,op右边的操作数对应第二个参数。 创建友元:
a、友元函数的原型应放在类声明中,并加上friend关键字
friend Time operator* (double n, const Time & t);
operator*在类中声明,但不是类的成员函数,但与成员函数具有相同的访问权限 b、在cpp文件中定义函数,由于不是成员函数,故不使用::限定符,也不要在定义中使用friend关键字 总结:类的友元函数是非成员函数,其访问权限与类成员函数相同。 3、常见的友元,重载<<操作符
重载<<操作符,使之能与cout一起来输出对象的内容 原型:
friend std::ostream & operator<< (std::ostream & os, const Time & t);
实现如下:
std::ostream & operator<< (std::ostream & os, const Time & t)
{
    os << t.m_hours << ":" << t.m_minutes;
    return os;
}
如此以来就可以直接输出Time对象
Time t(10.50)
cout << t; 一般来说,要重载<<操作符来显示c_name对象,可使用友元函数,定义如下:
ostream & operator<< (ostream & os, const & c_name obj)
{
  os << ....; //输出对象内容
  return os;
}
之所以要返回ostream的引用,主要作用是可以拼接输出,如 cout << "Current Time:" << t << "\n";

 

C++学习笔记-操作符重载的更多相关文章

  1. RxJava学习笔记(操作符)

    前言 上一篇文章介绍了RxJava的基础知识和简单实现,篇幅已经比较多了,所以把操作符(Operators)相关的内容放在这一篇.有了上一篇文章的基础,相信会比较容易理解操作符相关的内容了. 操作符( ...

  2. C++学习笔记-运算符重载

    运算符重载使得用户自定义的数据以一种更简洁的方式工作 运算符重载规则 重载运算符的限制 可以重载的运算符 + - * / % ^ & | ~ ! = < > += -= *= /= ...

  3. C++ Primer 学习笔记_60_重载操作符与转换 --赋值、下标、成员訪问操作符

    重载操作符与转换 --赋值.下标.成员訪问操作符 一.赋值操作符 类赋值操作符接受类类型形參,通常该形參是对类类型的const引用,但也能够是类类型或对类类型的非const引用.假设未定义这个操作符, ...

  4. C++ Primer 学习笔记_61_重载操作符与转换 --自增/自减操作符

    重载操作符与转换 --自增/自减操作符 引言: 自增,自减操作符常常由诸如迭代器这种类实现,这种类提供相似于指针的行为来訪问序列中的元素.比如,能够定义一个类,该类指向一个数组并为该数组中的元素提供訪 ...

  5. C++ Primer 学习笔记_62_重载操作符与转换 --调用操作符和函数对象

    重载操作符与转换 --调用操作符和函数对象 引言: 能够为类类型的对象重载函数调用操作符:一般为表示操作的类重载调用操作符! struct absInt { int operator() (int v ...

  6. C++ Primer 学习笔记_63_重载运算符和转换 --转换和类类型【上】

    重载运算符和转换 --转换与类类型[上] 引言: 在前面我们提到过:能够用一个实參调用的位 unsignedchar 相同范围的值,即:0到255. 这个类能够捕获下溢和上溢错误,因此使用起来比内置u ...

  7. 高放的c++学习笔记之重载运算与类型转换

    ▲基本概念 (1)重载运算符是具有特殊名字的函数,它们的名字又operator和其后要定义的运算符号共同构成.. (2)对于一个运算符号来说它或者是类的成员,或者至少含有一个类类型的参数. (3)我们 ...

  8. scala学习笔记——操作符

    中置操作符(二元操作符),操作符位于两个参数之间.操作符包括字母,比如to,也可以包括操作符字符,比如1->10,等同于方法调用1.->(10) a 标识符 b 其中的标识符是一个带有两个 ...

  9. C++学习笔记(5)----重载自增自减运算符

    自增运算符“++”和自减运算符“--”分别包含两个版本.即运算符前置形式(如 ++x)和运算符后置形式(如 x++),这两者进行的操作是不一样的.因此,当我们在对这两个运算符进行重载时,就必须区分前置 ...

随机推荐

  1. php -- 魔术方法 之 自动加载:__autoload()

    自动加载类 背景: 很多开发者写面向对象的应用程序时对每个类的定义建立一个 PHP 源文件.一个很大的烦恼是不得不在每个脚本开头写一个长长的包含文件列表(每个类一个文件). 在 PHP 5 中,不再需 ...

  2. Your Progress As A Programmer Is All Up To You

    Feb 3, 2014 I read a comment on a post on Hacker News where a young programmer said they didn't want ...

  3. ThinkPHP种where的使用(_logic and _complex)的使用实例

    1.对于thinkphp中的 and ,or 等复合型的查询,我要正确的使用相关的方法. a.实例 b.实例

  4. MyBitis(iBitis)系列随笔之一:MyBitis入门实例

        MyBits前身是iBitis,相对于Hibernate而言,它是半自动化ORM框架.本着分享和学习的目的,笔者将陆续把学习笔记与代码贴出,希望对想学习mybitis的同窗们有所帮助.     ...

  5. Struts2_day01--课程安排_Struts2概述_入门

    Struts2_day01 Struts2课程安排 今天内容 Struts2概述 Struts2框架入门 导入源文件 Struts2执行过程 查看源代码 Struts2的核心配置文件 标签packag ...

  6. HashMap实现原理、核心概念、关键问题的总结

    简单罗列一下较为重要的点: 同步的问题 碰撞处理问题 rehash的过程 put和get的处理过程 HashMap基础: HashMap的理论基础:维基百科哈希表 JDK中HashMap的描述:Has ...

  7. __init__()

    __init__() 是类的内置方法,用于初始化类的内部状态,当我们去实例化一个对象的时候,默认就会执行 __init__() 下面的语句,例子如下: #!/usr/bin/env python #- ...

  8. 《jquery权威指南2》学习笔记------基础函数

    Math.floor(Math.random() * 7 + 1); Math.random() 生成0和1之间的随机小数Math.random() * 7 生成0和7之间的随机小数Math.rand ...

  9. CentOS7 minimal下MySQL安装

    http://www.linuxidc.com/Linux/2016-12/137942.htm 首先要使用root用户登录 卸载: 1.卸载原有程序 yum remove mysql mysql-s ...

  10. Objective-C入门教材

    2011-05-11 15:58 三聪 cnblogs 字号:T | T 阅读本文前,你也要了解面向对象的基本概念.对象的使用以及面象对象设计模式都是bjective-C进行面向对象编程和设计Coco ...