在C ++中,我们可以使运算符适用于用户定义的类。 这意味着C ++能够为运算符提供数据类型的特殊含义,这种能力称为运算符重载。

例如,我们可以在像String这样的类中重载运算符'+',这样我们就可以通过使用+来连接两个字符串。

其它示例中算术运算符可以重载的的类是复数,小数,大整数等。

运算符重载的语法格式:

Return_Type classname :: operator op(Argument list)

{

Function Body

}

在上面的语法中,Return_Type是要返回给另一个对象的值类型,operator op是运算符是关键字的函数,op是要重载的运算符。

运算符函数必须是非静态(成员函数)或友元函数。

运算符重载可以通过三种方法完成,它们是

1)      重载一元运算符。

2)      重载二元运算符。

3)      使用友元函数重载二元运算符。

以下是定义运算符函数的一些条件/规则:

l  在非静态函数的情况下,二元运算符应该只有一个参数,而一元不应该有一个参数。

l  在友元函数的情况下,二元运算符应该只有两个参数,而一元应该只有一个参数。

l  如果实现了运算符重载,则所有类成员对象都应该是公共的。

运算符函数和普通函数有什么区别?

运算符函数与普通函数相同。 唯一的区别是,运算符函数的名称始终是operator关键字,后跟运算符符号,并且在使用相应的运算符时调用运算符函数。

除了极个别的运算符外,大部分运算符都可以被重载。

以下是可以重载的运算符列表:

以下是不可以重载的运算符列表:

为什么(点),::,?:和sizeof不可以重载看这里解释。

关于运算符重载的重要方面:

1)为了使操作符重载起作用,其中一个操作数必须是用户定义的类对象。

2)赋值运算符:编译器自动为每个类创建一个默认赋值运算符。 默认赋值运算符确实将右侧的所有成员分配到左侧,并且在大多数情况下都能正常工作(此行为与复制构造函数相同)。

3)转换运算符:我们还可以编写可用于将一种类型转换为另一种类型的转换运算符。重载的转换运算符必须是成员方法。 其他运算符可以是成员方法或全局方法。

4)任何可以用单个参数调用的构造函数都可以用作转换构造函数,这意味着它也可以用于隐式转换为正在构造的类。

综合例子:学习测试,仅供参考!

OperatorOverloadingClass.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
 
#ifndef OPERATOROVERLOADINGCLASS_H
#define OPERATOROVERLOADINGCLASS_H

#include <iostream>
using namespace std;

;

class OperatorOverloadingClass
{
public:
    OperatorOverloadingClass();
    OperatorOverloadingClass(const int &number,
                             const float &fnumber);

// 赋值运算符
    void operator = (const OperatorOverloadingClass &obj)
    {
        m_number = obj.m_number;
    }

// 函数调用()运算符
    OperatorOverloadingClass operator()(int a, int b, int c)
    {
        OperatorOverloadingClass D;
        // just put random calculation
        D.m_number = a + b + c;
        return D;
    }

void print();

/* OperatorOverloading
    */
    // 算术运算符
    // +
    OperatorOverloadingClass operator + (const OperatorOverloadingClass &obj);
    // -
    OperatorOverloadingClass operator - (const OperatorOverloadingClass &obj);
    // *
    OperatorOverloadingClass operator * (const OperatorOverloadingClass &obj);
    // /
    OperatorOverloadingClass operator / (const OperatorOverloadingClass &obj);
    // %
    OperatorOverloadingClass operator % (const OperatorOverloadingClass &obj);

// 关系运算符
    // <
    bool operator < (const OperatorOverloadingClass &obj);
    // >
    bool operator > (const OperatorOverloadingClass &obj);
    // <=
    bool operator <= (const OperatorOverloadingClass &obj);
    // >=
    bool operator >= (const OperatorOverloadingClass &obj);
    // ==
    bool operator == (const OperatorOverloadingClass &obj);
    // !=
    bool operator != (const OperatorOverloadingClass &obj);

// 位运算符
    // ^
    OperatorOverloadingClass operator ^ (const OperatorOverloadingClass &obj);
    // &
    OperatorOverloadingClass operator & (const OperatorOverloadingClass &obj);
    // |
    OperatorOverloadingClass operator | (const OperatorOverloadingClass &obj);

// 逻辑运算符
    // &&
    bool operator && (const OperatorOverloadingClass &obj);
    // ||
    bool operator || (const OperatorOverloadingClass &obj);

// 取地址运算符&
    int *operator &()
    {
        return &this->m_number;
    }
    // 内存操作运算符*
    int operator *(OperatorOverloadingClass *obj)
    {
        return obj->m_number;
    }

// 输入输出运算符
    // <<
    friend ostream &operator<<(ostream &output, const OperatorOverloadingClass &obj )
    {
        output << "number : " << obj.m_number;
        return output;
    }
    // >>
    friend istream &operator>>(istream &input, OperatorOverloadingClass &obj)
    {
        input >> obj.m_number;
        return input;
    }

// 自增自减运算符
    // ++i
    OperatorOverloadingClass operator ++();
    // i++
    OperatorOverloadingClass operator ++(int);
    // --i
    OperatorOverloadingClass operator --();
    // i--
    OperatorOverloadingClass operator --(int);

// 数组成员访问运算符[]
    int &operator[](int i)
    {
        if( i > SIZE )
        {
            cout << "Index out of bounds" << endl;
            // return first element.
];
        }
        return arr[i];
    }

// 类成员访问运算符->
    // https://www.cnblogs.com/codingmengmeng/p/9064702.html

// 复合赋值运算符:+=、-=、*=、/=、%=、<<=、>>=、^=、&=、|=
    void operator += (const OperatorOverloadingClass &obj)
    {
        m_number += obj.m_number;
    }
    // 其余类似

// 重载类型转换
    operator int()
    {
        return m_number;
    }
    operator float()
    {
        return m_fnumber;
    }

// new delete
    void *operator new(size_t size);
    void operator delete(void *p);
    void *operator new[](size_t size);
    void operator delete[](void *p, size_t size);

private:
    int     m_number;
    float   m_fnumber;
    int     arr[SIZE];
};

#endif // OPERATOROVERLOADINGCLASS_H

 OperatorOverloadingClass.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
 
#include "OperatorOverloadingClass.h"
#include <QDebug>

OperatorOverloadingClass::OperatorOverloadingClass()
{
    m_number = ;
    m_fnumber = ;
    ; i < SIZE; i++)
    {
        arr[i] = i;
    }
}

OperatorOverloadingClass::OperatorOverloadingClass(const int &number,
        const float &fnumber)
    : m_number(number)
    , m_fnumber(fnumber)
{
    ; i < SIZE; i++)
    {
        arr[i] = i;
    }
}

void OperatorOverloadingClass::print()
{
    qDebug() << "m_number =" << m_number;
}

OperatorOverloadingClass OperatorOverloadingClass::operator +(const OperatorOverloadingClass &obj)
{
    OperatorOverloadingClass out;
    out.m_number = this->m_number + obj.m_number;
    return out;
}

OperatorOverloadingClass OperatorOverloadingClass::operator -(const OperatorOverloadingClass &obj)
{
    OperatorOverloadingClass out;
    out.m_number = this->m_number - obj.m_number;
    return out;
}

OperatorOverloadingClass OperatorOverloadingClass::operator *(const OperatorOverloadingClass &obj)
{
    OperatorOverloadingClass out;
    out.m_number = this->m_number * obj.m_number;
    return out;
}

OperatorOverloadingClass OperatorOverloadingClass::operator /(const OperatorOverloadingClass &obj)
{
    OperatorOverloadingClass out;
    out.m_number = this->m_number / obj.m_number;
    return out;
}

OperatorOverloadingClass OperatorOverloadingClass::operator %(const OperatorOverloadingClass &obj)
{
    OperatorOverloadingClass out;
    out.m_number = this->m_number % obj.m_number;
    return out;
}

OperatorOverloadingClass OperatorOverloadingClass::operator ^(const OperatorOverloadingClass &obj)
{
    OperatorOverloadingClass out;
    out.m_number = (this->m_number) ^ (obj.m_number);
    return out;
}

OperatorOverloadingClass OperatorOverloadingClass::operator &(const OperatorOverloadingClass &obj)
{
    OperatorOverloadingClass out;
    out.m_number = (this->m_number) & (obj.m_number);
    return out;
}

OperatorOverloadingClass OperatorOverloadingClass::operator |(const OperatorOverloadingClass &obj)
{
    OperatorOverloadingClass out;
    out.m_number = (this->m_number) | (obj.m_number);
    return out;
}

bool OperatorOverloadingClass::operator &&(const OperatorOverloadingClass &obj)
{
    return (this->m_number) && (obj.m_number);
}

bool OperatorOverloadingClass::operator ||(const OperatorOverloadingClass &obj)
{
    return (this->m_number) || (obj.m_number);
}

bool OperatorOverloadingClass::operator <(const OperatorOverloadingClass &obj)
{
    if(m_number < obj.m_number)
    {
        return true;
    }
    return false;
}

bool OperatorOverloadingClass::operator >(const OperatorOverloadingClass &obj)
{
    if(m_number > obj.m_number)
    {
        return true;
    }
    return false;
}

bool OperatorOverloadingClass::operator <=(const OperatorOverloadingClass &obj)
{
    if(m_number <= obj.m_number)
    {
        return true;
    }
    return false;
}

bool OperatorOverloadingClass::operator >=(const OperatorOverloadingClass &obj)
{
    if(m_number >= obj.m_number)
    {
        return true;
    }
    return false;
}

bool OperatorOverloadingClass::operator ==(const OperatorOverloadingClass &obj)
{
    if(m_number == obj.m_number)
    {
        return true;
    }
    return false;
}

bool OperatorOverloadingClass::operator !=(const OperatorOverloadingClass &obj)
{
    if(m_number != obj.m_number)
    {
        return true;
    }
    return false;
}

// overloaded prefix ++ operator
OperatorOverloadingClass OperatorOverloadingClass:: operator++ ()
{
    ++m_number;          // increment this object
);
}

// overloaded postfix ++ operator
OperatorOverloadingClass OperatorOverloadingClass:: operator++( int )
{

// save the orignal value
);

// increment this object
    ++m_number;

// return old original value
    return T;
}

// overloaded prefix -- operator
OperatorOverloadingClass OperatorOverloadingClass:: operator-- ()
{
    --m_number;          // increment this object
);
}

// overloaded postfix -- operator
OperatorOverloadingClass OperatorOverloadingClass:: operator--( int )
{

// save the orignal value
);

// increment this object
    --m_number;

// return old original value
    return T;
}

void *OperatorOverloadingClass::operator new(size_t size)
{
    return malloc(size);
}

void OperatorOverloadingClass::operator delete(void *p)
{
    free(p);
}

inline void *OperatorOverloadingClass::operator new[](size_t size)
{
    OperatorOverloadingClass *p = (OperatorOverloadingClass *)malloc(size);
    return p;
}

inline void OperatorOverloadingClass::operator delete[](void *p, size_t size)
{
    free(p);
}

C++运算符重载学习总结的更多相关文章

  1. 初步C++运算符重载学习笔记&lt;3&gt; 增量递减运算符重载

    初步C++运算符重载学习笔记<1> 初探C++运算符重载学习笔记<2> 重载为友元函数     增量.减量运算符++(--)分别有两种形式:前自增++i(自减--i).后自增i ...

  2. 初探C++运算符重载学习笔记&lt;2&gt; 重载为友元函数

    初探C++运算符重载学习笔记 在上面那篇博客中,写了将运算符重载为普通函数或类的成员函数这两种情况. 以下的两种情况发生.则我们须要将运算符重载为类的友元函数 <1>成员函数不能满足要求 ...

  3. C++学习笔记之运算符重载

    一.运算符重载基本知识 在前面的一篇博文 C++学习笔记之模板(1)——从函数重载到函数模板 中,介绍了函数重载的概念,定义及用法,函数重载(也被称之为函数多态)就是使用户能够定义多个名称相同但特征标 ...

  4. C++学习之运算符重载的总结

    C++学习之运算符重载的总结              运算符重载是对已有的运算符赋予多重含义,使同一个运算符作用域不同类型的数据导致不同行为的发生,C++为运算符重载提供了一种方法,即运算符重载函数 ...

  5. c++中的运算符重载operator2(翁恺c++公开课[31-33]学习笔记)

    上一篇operator1中,大概说了下重载的基本用法,接下来对c++中常见的可重载运算符归一下类,说一下它们的返回值,讨论下较为复杂的运算符重载上的坑

  6. c++中的运算符重载operator1(翁恺c++公开课[30]学习笔记)

    运算符重载规则: 只有已经存在的运算符才能被重载,不能自己制造一个c++中没有的运算符进行重载 重载可以在类或枚举类型内进行,也可以是全局函数,但int.float这种已有的类型内是不被允许的 不能二 ...

  7. C++基础 学习笔记五:重载之运算符重载

    C++基础 学习笔记五:重载之运算符重载 什么是运算符重载 用同一个运算符完成不同的功能即同一个运算符可以有不同的功能的方法叫做运算符重载.运算符重载是静态多态性的体现. 运算符重载的规则 重载公式 ...

  8. 我的c++学习(8)运算符重载和友元

    运算符的重载,实际是一种特殊的函数重载,必须定义一个函数,并告诉C++编译器,当遇到该运算符时就调用此函数来行使运算符功能.这个函数叫做运算符重载函数(常为类的成员函数). 方法与解释 ◆ 1.定义运 ...

  9. C++学习26 运算符重载的概念和语法

    所谓重载,就是赋予新的含义.函数重载(Function Overloading)可以让一个函数名有多种功能,在不同情况下进行不同的操作.运算符重载(Operator Overloading)也是一个道 ...

随机推荐

  1. 每天一套题打卡|河南省第七届ACM/ICPC

    A 海岛争霸 题目:Q次询问,他想知道从岛屿A 到岛屿B 有没有行驶航线,若有的话,所经过的航线,危险程度最小可能是多少. 多源点最短路,用floyd 在松弛更新:g[i][k] < g[i][ ...

  2. DNS解析过程--笔试答题版

    在运维笔试的时候,回答DNS解析的过程,不能写一大堆,一是不美观,二是浪费时间,应该怎么写呢?我觉得这样写比较好. 1.客户端:chche----hosts 2.DNS服务器:cache---递归-- ...

  3. 201871010101-陈来弟《面向对象程序设计(Java)》第八周学习总结

    实验七 接口的定义与使用 第一部分:理论知识 一.接口.lambda和内部类:  Comparator与comparable接口: 1.comparable接口的方法是compareTo,只有一个参数 ...

  4. 201871010128-杨丽霞《面向对象程序设计(Java)》第十一周学习总结

    201871010128-杨丽霞<面向对象程序设计(Java)>第十一周学习总结 项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ ...

  5. 201871010134-周英杰《面向对象程序设计(java)》第十六周学习总结

    项目 内容 <面向对象程序设计(java)> https://www.cnblogs.com/nwnu-daizh/ 这个作业的要求在哪里 https://www.cnblogs.com/ ...

  6. eclipse配置JDK和设置编译版本的几种方法

    eclipse配置JDK和设置编译版本的几种方法  2016-12-13 16:08             http://blog.csdn.net/gnail_oug/article/detail ...

  7. 各大开源rpc 框架 比较

    各大开源rpc 框架 比较   1. 前言 随着现在互联网行业的发展,越来越多的框架.中间件.容器等开源技术不断地涌现,更好地来服务于业务,解决实现业务的问题.然而面对众多的技术选择,我们要如何甄别出 ...

  8. python3对urllib和urllib2进行了重构

    python3对urllib和urllib2进行了重构,拆分成了urllib.request,urllib.response, urllib.parse, urllib.error等几个子模块,这样的 ...

  9. vue和react的区别

    数据: vue:双向数据绑定和单向数据流.双向数据绑定:DOM元素绑定的data值,当发生改变后,vue的响应式机制会自动监听data的变化重新渲染.单向数据流:当父组件给子组件传递数据的时候,子组件 ...

  10. Centos7下搭建NFS服务器与连接详解

    一,环境介绍    本实验使用了两台centos7虚拟机,其中         服务器:192.168.1.188    客户端:192.168.1.189 二,实验步骤    192.168.1.1 ...