C++回顾day02---<运算符重载>
一:运算符重载的限制
(一)可以重载的运算符:
+ - * / % ^ & | ~
! = < > += -= *= /= %=
^= &= |= << >> <<= >>= == !=
<= >= && || ++ -- , ->* ->
() [] new new[] delete delete[]
(二)不允许重载的运算符:
(三)原有基本语义不变
1.不改变运算符的优先级
运算符
|
描述
|
例子
|
可重载性
|
第一级别
|
|||
::
|
作用域解析符
|
Class::age = 2;
|
不可重载
|
第二级别
|
|||
()
|
函数调用
|
isdigit('1')
|
可重载
|
()
|
成员初始化
|
c_tor(int x, int y) : _x(x), _y(y*10){};
|
可重载
|
[]
|
数组数据获取
|
array[4] = 2;
|
可重载
|
->
|
指针型成员调用
|
ptr->age = 34;
|
可重载
|
.
|
对象型成员调用
|
obj.age = 34;
|
不可重载
|
++
|
后自增运算符
|
for( int i = 0; i < 10; i++ ) cout << i;
|
可重载
|
--
|
后自减运算符
|
for( int i = 10; i > 0; i-- ) cout << i;
|
可重载
|
const_cast
|
特殊属性转换
|
const_cast<type_to>(type_from);
|
不可重载
|
dynamic_cast
|
特殊属性转换
|
dynamic_cast<type_to>(type_from);
|
不可重载
|
static_cast
|
特殊属性转换
|
static_cast<type_to>(type_from);
|
不可重载
|
reinterpret_cast
|
特殊属性转换
|
reinterpret_cast<type_to>(type_from);
|
不可重载
|
typeid
|
对象类型符
|
cout « typeid(var).name();
cout « typeid(type).name();
|
不可重载
|
第三级别(具有右结合性)
|
|||
!
|
逻辑取反
|
if( !done ) …
|
可重载
|
not
|
! 的另一种表达
|
||
~
|
按位取反
|
flags = ~flags;
|
可重载
|
compl
|
~的另一种表达
|
||
++
|
预自增运算符
|
for( i = 0; i < 10; ++i ) cout << i;
|
可重载
|
--
|
预自减运算符
|
for( i = 10; i > 0; --i ) cout << i;
|
可重载
|
-
|
负号
|
int i = -1;
|
可重载
|
+
|
正号
|
int i = +1;
|
可重载
|
*
|
指针取值
|
int data = *intPtr;
|
可重载
|
&
|
值取指针
|
int *intPtr = &data;
|
可重载
|
new
|
动态元素内存分配
|
long *pVar = new long;
MyClass *ptr = new MyClass(args);
|
可重载
|
new []
|
动态数组内存分配
|
long *array = new long[n];
|
可重载
|
delete
|
动态析构元素内存
|
delete pVar;
|
可重载
|
delete []
|
动态析构数组内存
|
delete [] array;
|
可重载
|
(type)
|
强制类型转换
|
int i = (int) floatNum;
|
可重载
|
sizeof
|
返回类型内存
|
int size = sizeof floatNum;
int size = sizeof(float);
|
不可重载
|
第四级别
|
|||
->*
|
类指针成员引用
|
ptr->*var = 24;
|
可重载
|
.*
|
类对象成员引用
|
obj.*var = 24;
|
不可重载
|
第五级别
|
|||
*
|
乘法
|
int i = 2 * 4;
|
可重载
|
/
|
除法
|
float f = 10.0 / 3.0;
|
可重载
|
%
|
取余数(模运算)
|
int rem = 4 % 3;
|
可重载
|
第六级别
|
|||
+
|
加法
|
int i = 2 + 3;
|
可重载
|
-
|
减法
|
int i = 5 - 1;
|
可重载
|
第七级别
|
|||
<<
|
位左移
|
int flags = 33 << 1;
|
可重载
|
>>
|
位右移
|
int flags = 33 >> 1;
|
可重载
|
第八级别
|
|||
<
|
小于
|
if( i < 42 ) …
|
可重载
|
<=
|
小于等于
|
if( i <= 42 ) ...
|
可重载
|
>
|
大于
|
if( i > 42 ) …
|
可重载
|
>=
|
大于等于
|
if( i >= 42 ) ...
|
可重载
|
第九级别
|
|||
==
|
恒等于
|
if( i == 42 ) ...
|
可重载
|
eq
|
== 的另一种表达
|
||
!=
|
不等于
|
if( i != 42 ) …
|
可重载
|
not_eq
|
!=的另一种表达
|
||
第十级别
|
|||
&
|
位且运算
|
flags = flags & 42;
|
可重载
|
bitand
|
&的另一种表达
|
||
第十一级别
|
|||
^
|
位异或运算
|
flags = flags ^ 42;
|
可重载
|
xor
|
^的另一种表达
|
||
第十二级别
|
|||
|
|
位或运算
|
flags = flags | 42;
|
可重载
|
bitor
|
|的另一种表达
|
||
第十三级别
|
|||
&&
|
逻辑且运算
|
if( conditionA && conditionB ) …
|
可重载
|
and
|
&&的另一种表达
|
||
第十四级别
|
|||
||
|
逻辑或运算
|
if( conditionA || conditionB ) ...
|
可重载
|
or
|
||的另一种表达
|
||
第十五级别(具有右结合性)
|
|||
? :
|
条件运算符
|
int i = (a > b) ? a : b;
|
不可重载
|
第十六级别(具有右结合性)
|
|||
=
|
赋值
|
int a = b;
|
可重载
|
+=
|
加赋值运算
|
a += 3;
|
可重载
|
-=
|
减赋值运算
|
b -= 4;
|
可重载
|
*=
|
乘赋值运算
|
a *= 5;
|
可重载
|
/=
|
除赋值运算
|
a /= 2;
|
可重载
|
%=
|
模赋值运算
|
a %= 3;
|
可重载
|
&=
|
位且赋值运算
|
flags &= new_flags;
|
可重载
|
and_eq
|
&= 的另一种表达
|
||
^=
|
位异或赋值运算
|
flags ^= new_flags;
|
可重载
|
xor_eq
|
^=的另一种表达
|
||
|=
|
位或赋值运算
|
flags |= new_flags;
|
可重载
|
or_eq
|
|=的另一种表达
|
||
<<=
|
位左移赋值运算
|
flags <<= 2;
|
可重载
|
>>=
|
位右移赋值运算
|
flags >>= 2;
|
可重载
|
第十七级别
|
|||
throw
|
异常抛出
|
throw EClass(“Message”);
|
不可重载
|
第十八级别
|
|||
,
|
逗号分隔符
|
for( i = 0, j = 0; i < 10; i++, j++ ) …
|
可重载
|
2.不改变运算符的结合性
例如:=结合性是从右向左
a=b=;
先执行b=,再执行a=b
3.不改变运算符所需要的操作数
+-*/等需要两个操作数
++ --等一个操作数
.....
4.不能创建新的运算符
二:运算符重载两种方法(传参方式不同,实现代码不同,应用场合不同)
(一)成员函数进行运算符重载(有this指针,所以传参少一个)
class A
{
private:
int op;
public:
A(int a1) :op(a1)
{
} A& operator+(const A& a) //二元运算符,第一数为this对象,第二个是a对象
{
A nonObj(this->op + a.op);
return nonObj; //返回匿名对象
} void getInfo()
{
cout << "op:" << op << endl;
}
}; void main()
{
A a1();
A a2();
A a3 = a1 + a2;
a3.getInfo();
system("pause");
}
或者A& A::operator+(const A& a){}
(二)友元函数进行运算符重载(无this指针)
class A
{
friend A& operator+(const A& a, const A& b); //补充:默认访问修饰符为private
private:
int op;
public:
A(int a1) :op(a1)
{
} void getInfo()
{
cout << "op:" << op << endl;
}
}; A& operator+(const A& a, const A& b) //私有是对于类,友元函数可以使用类的所有成员,不管访问修饰符
{
A nonObj(a.op + b.op);
return nonObj;
} void main()
{
A a1();
A a2();
A a3 = a1 + a2;
a3.getInfo();
system("pause");
}
(三)上面实现的是二元运算符重载,这里来实现一元运算符重载(区分符号前置后置)
友元函数和成员函数操作运算符需要区分前置和后置(符号前后置)
前置 operator++()
++a
后置 operator++(int) //int进行占位区分 后面多一个,为后置
a++
class A
{
friend A& operator++(const A& a);
private:
int op;
public:
A(int a1) :op(a1)
{
} A& operator--() //无伪参数占位,是前置--a
{
this->op--;
A nonObj(this->op);
return nonObj;
} A& operator--(int) //使用int伪参数占位,为后置a--
{
A nonObj(this->op);
this->op--;
return nonObj;
} void getInfo()
{
cout << "op:" << op << endl;
}
}; void main()
{
A a1();
A a2();
cout << "a1";
a1.getInfo();
A a4 = a1--;
cout << "a4";
a4.getInfo();
cout << "a1";
a1.getInfo(); cout <<"--------"<< endl; cout << "a2";
a2.getInfo();
A a5 = --a2;
cout << "a5";
a5.getInfo();
cout << "a2";
a2.getInfo(); system("pause");
}
成员函数重载操作符(前置后置区分)
class A
{
friend A& operator++(A& a);
friend A& operator++(A& a,int);
private:
int op;
public:
A(int a1) :op(a1)
{
} void getInfo()
{
cout << "op:" << op << endl;
}
}; A& operator++(A& a) //对于一目运算符不能设置为const,因为要对自己进行修改
{
a.op++;
A nonObj(a.op);
return nonObj;
} A& operator++(A& a, int)
{
A nonObj(a.op);
a.op++;
return nonObj;
} void main()
{
A a1();
A a2();
cout << "a1";
a1.getInfo();
A a4 = a1++;
cout << "a4";
a4.getInfo();
cout << "a1";
a1.getInfo(); cout <<"--------"<< endl; cout << "a2";
a2.getInfo();
A a5 = ++a2;
cout << "a5";
a5.getInfo();
cout << "a2";
a2.getInfo(); system("pause");
}
友元函数重载操作符(前置后置区别)
三:运算符重载应用场景
(一)友元函数和成员函数选择方法
1.当无法修改左操作数的类是,使用全局友元函数进行重载
对<<或者>>重载
cout<<数
cin>>数
左操作都是ostream或者istream无法修改的类,
所以重载<<或者>>只能用全局友元函数
2.=,[],(),->操作符只能使用成员函数进行重载
(二)使用友元函数方法实现<<和>>操作符重载
注意:
<<返回ostream类型
cout<<a<<b //连续输出a b两个变量
>>返回istream类型
cin>>a>>b; //连续输入两个变量到a b
实现:
class A
{
friend ostream& operator<<(ostream& out, A& a);
friend istream& operator>>(istream& in, A& a);
private:
int op;
public:
A(int a1) :op(a1)
{
} void getInfo()
{
cout << "op:" << op << endl;
}
}; ostream& operator<<(ostream& out, A& a)
{
out <<"overload<<:"<< a.op << endl;
return out;
} istream& operator>>(istream& in, A& a)
{
in >> a.op;
return in;
} void main()
{
A a();
cout << a; cin >> a;
cout << a; system("pause");
}
(三)使用成员函数方法实现=,数组下标运算符[],函数调用符(),操作符重载
1.=和[]
class A
{
friend ostream& operator<<(ostream& out, A& a);
friend istream& operator>>(istream& in, A& a);
private:
int op;
int *pArr;
public:
A(int a1) :op(a1)
{
pArr = (int *)malloc(sizeof(int)*a1);
for (int i = ; i < a1;i++)
{
pArr[i] = i+;
}
} void getInfo()
{
cout << "op:" << op << endl;
} //赋值运算符
A& operator=(const A& a1)
{
cout << "operator=" << endl;
A nonObj(this->op + a1.op);
return nonObj;
} //数组下标运算符
int& operator[](int i)
{
cout << "operator[]" << endl;
return this->pArr[i];
} ~A()
{
if (this->pArr != NULL)
free(this->pArr);
}
}; ostream& operator<<(ostream& out, A& a)
{
out <<"overload<<:"<< a.op << endl;
return out;
} istream& operator>>(istream& in, A& a)
{
in >> a.op;
return in;
} void main()
{
A a();
//A b = a; 是错误的,不会调用=重载操作符,是初始化操作
A b();
b = a; //会调用=重载运算符
cout << a[]; system("pause");
}
2.函数调用符():用于函数调用,实现数学函数的抽象
class Func
{
public:
double operator()(double x, double y); //第一个()是重载的符号
}; double Func::operator()(double x, double y)
{
return x*x + y*y;
} void main()
{
Func func; //调用默认构造函数
double ret = func(3.0, 4.0);
cout << ret << endl;
system("pause");
}
(四)&&和||虽然可以实现重载,但是由于重载后我们无法实现短路规则(还是要靠&& ||),所以最好不要重载
C++回顾day02---<运算符重载>的更多相关文章
- c++入门之—运算符重载和友元函数
运算符重载的意义是:将常见的运算符重载出其他的含义:比如将*重载出指针的含义,将<<与cout联合使用重载出输出的含义,但需要认识到的问题是:运算符的重载:本质仍然是成员函数,即你可以认为 ...
- Swift教程之运算符重载
http://blog.csdn.net/mengxiangyue/article/details/43437797 原文地址:http://www.raywenderlich.com/80818/o ...
- C++ 运算符重载时,将运算符两边对象交换问题.
在C++进行运算符重载时, 一般来讲,运算符两边的对象的顺序是不能交换的. 比如下面的例子: #include <iostream> using namespace std; class ...
- C#高级编程笔记2016年10月12日 运算符重载
1.运算符重载:运算符重重载的关键是在对象上不能总是只调用方法或属性,有时还需要做一些其他工作,例如,对数值进行相加.相乘或逻辑操作等.例如,语句if(a==b).对于类,这个语句在默认状态下会比较引 ...
- C++运算符重载
C++运算符重载 基本知识 重载的运算符是具有特殊名字的函数,他们的名字由关键字operator和其后要定义的运算符号共同组成. 运算符可以重载为成员函数和非成员函数.当一个重载的运算符是成员函数时, ...
- 标准C++之运算符重载和虚表指针
1 -> *运算符重载 //autoptr.cpp #include<iostream> #include<string> using namespace std ...
- python运算符重载
python运算符重载就是在解释器使用对象内置操作前,拦截该操作,使用自己写的重载方法. 重载方法:__init__为构造函数,__sub__为减法表达式 class Number: def __in ...
- PoEduo - C++阶段班【Po学校】-Lesson03-5_运算符重载- 第7天
PoEduo - Lesson03-5_运算符重载- 第7天 复习前面的知识点 空类会自动生成哪些默认函数 6个默认函数 1 构造 2 析构 3 赋值 4 拷贝构造 5 oper ...
- 不可或缺 Windows Native (24) - C++: 运算符重载, 自定义类型转换
[源码下载] 不可或缺 Windows Native (24) - C++: 运算符重载, 自定义类型转换 作者:webabcd 介绍不可或缺 Windows Native 之 C++ 运算符重载 自 ...
- 我的c++学习(8)运算符重载和友元
运算符的重载,实际是一种特殊的函数重载,必须定义一个函数,并告诉C++编译器,当遇到该运算符时就调用此函数来行使运算符功能.这个函数叫做运算符重载函数(常为类的成员函数). 方法与解释 ◆ 1.定义运 ...
随机推荐
- 启动期间的内存管理之pagging_init初始化分页机制--Linux内存管理(十四)
1 今日内容(分页机制初始化) 在初始化内存的结点和内存区域之前, 内核先通过pagging_init初始化了内核的分页机制. 在分页机制完成后, 才会开始初始化系统的内存数据结构(包括内存节点数据和 ...
- Kafka集成Kerberos之后如何使用生产者消费者命令
1.生产者1.1.准备jaas.conf并添加到环境变量(使用以下方式的其中一种)1.1.1.使用Kinit方式前提是手动kinit 配置内容为: KafkaClient { com.sun.secu ...
- 3.20 总结 java程序流程控制
- Mysql共享锁、排他锁、悲观锁、乐观锁及其使用场景
一.相关名词 |--表级锁(锁定整个表) |--页级锁(锁定一页) |--行级锁(锁定一行) |--共享锁(S锁,MyISAM 叫做读锁) |--排他锁(X锁,MyISAM 叫做写锁) |--悲观锁( ...
- B. Obtaining the String(模拟)
比较水的模拟 思路:就是模拟题意 注意:把数组开大点,开始wa了几次就是这个原因 #include<iostream> #include<string> #include< ...
- SQL学习 DECODE
from 百度百科: DECODE有什么用途呢? 先构造一个例子,假设我们想给这些职员加工资,其标准是:工资在8000元以下的加20%:工资在8000元或以上的加15%,通常的做法是,先选出记录中的工 ...
- 在区块链上表白——使用C#将一句话放入比特币的区块链上
最近在看区块链和比特币的知识,顺便简单研究了一下BitCoin的脚本语言,发现OP_RETURN这个命令可以在后面放入自己想说的内容,很多侧链啊,公证之类就是利用了这个特性,可以把一句话,或者一个哈希 ...
- 安装Laravel框架,利用composer
学一学PHP框架--Laravel的设计思想. 先安装Laravel: Laravel的文档很全:参考 http://www.golaravel.com/ 既然文档很全,就简单说下几个重点.以下以安装 ...
- 在Bootstrap开发框架的工作流模块中实现流程完成后更新资料状态处理
在开发查看流程表单明细的时候,在Web界面中,我们往往通过使用@RenderPage实现页面内容模块化的隔离,减少复杂度,因此把一些常用的如审批.撤销.会签.阅办等等的流程步骤都放到了通用处理的页面V ...
- Python--day04(流程控制)
day03主要内容回顾 1.变量名命名规范 -- 1.只能由数字.字母 及 _ 组成 -- 2.不能以数字开头 -- 3.不能与系统关键字重名 -- 4._开头有特殊含义 -- 5.__开头__结尾的 ...