C++ 运算符重载的基本概念

简介: C++ 运算符重载的基本概念

1.运算符重载的需求


C++ 预定义的运算符,只能用于基本数据类型的运算:整型、实型、字符型、逻辑型等等,且不能用于对象的运算。但是我们有时候又很需要在对象之间能用运算符,那么这时我们就要重载运算符,使得运算符能用于对象之间的运算。


比如,在数学上,两个复数可以直接进行+、-等运算,但在C++中,直接将+或-用于复数对象是不允许的。有时会希望,让对象也能通过运算符进行运算。这样代码就更简洁,也容易理解。


例如:

complex_a 和 complex_b 是两个复数对象,求两个复数的和,希望的能直接写成:complex_a + complex_b

这时我们就需要对+号运算符进行重载。



2.运算符重载的形式


运算符重载的实质就是函数重载,可以重载为普通函数,也可以重载为成员函数运算符重载的基本形式如下:

返回值类型operator运算符(形参表)

{

 ...

}


下面举个例子,实现对复数对象的+-运算符重载:

classComplex// 复数类

{

public:

// 构造函数,如果不传参数,默认把实部和虚部初始化为0

Complex(doubler=0.0, doublei=0.0):m_real(r),m_imag(i) { }


// 重载-号运算符,属于成员函数

Complexoperator-(constComplex&c)

{

// 返回一个临时对象

returnComplex(m_real-c.m_real, m_imag-c.m_imag);

}


// 打印复数

voidPrintComplex()

{

cout<<m_real<<","<<m_imag<<endl;

}

 

  // 将重载+号的普通函数,定义成友元函数

  // 目的是为了友元函数能访问对象的私有成员

  friendComplexoperator+(constComplex&a, constComplex&b);

 

private:

doublem_real;  // 实部的值

doublem_imag;  // 虚部的值

};


// 重载+号运算符,属于普通函数,不是对象的成员函数

Complexoperator+(constComplex&a, constComplex&b)

{

// 返回一个临时对象

returnComplex(a.m_real+b.m_real, a.m_imag+b.m_imag);

}


intmain()

{

Complexa(2,2);

Complexb(1,1);

Complexc;


c=a+b; // 等价于c = operator+(a,b)

c.PrintComplex();


c=a-b; // 等价于 c = a.operator-(b)

c.PrintComplex();


return0;

}

输出结果:

3,3

1,1


从上面的例子中,我们可以知道重载为成员函数和普通函数的区别了:

  • 重载为成员函数时,参数个数为运算符目数减一。如:c = a - b;等价于c = a.operator-(b)
  • 重载为普通函数时,参数个数为运算符目数。如:c = a + b; 等价于c = operator+(a,b)


在上面的代码中,我把重载+号运算符的普通函数,在Complex复数类中定义成了友元函数,目的是为了友元函数能访问对象的私有成,否则会编译报错。


3.加减运算符的返回值和参数表


这里还有个值得思考的问题:

  • 为什么重载-号和+号运算符函数的返回类型是Complex对象而不是Complex &呢?
  • 为什么重载-号和+号的运算符函数的参数表是const Complex & c常引用类型而不是Complex c呢?

// 重载-号运算符,属于成员函数

ComplexComplex::operator-(constComplex&c)

{

// 返回一个临时对象

returnComplex(m_real-c.m_real, m_imag-c.m_imag);

}


首先先说一下参数表为什么是const Complex & c常引用类型,首先如果参数表如果普通的对象形式Complex c,那么在入参的时候,就会调用默认的赋值(拷贝)构造函数,产生了一个临时对象,这会增大开销,所以就采用引用的方式,同时又为了防止引用的对象被修改,所以就定义成了const Complex & c常引用类型。


再来说一下返回值为什么是普通Complex对象,因为本次 - 号和 + 号运算符的函数执行之后,需要返回一个新的对象给到左值。

相关文章
|
1月前
|
程序员 C++ 开发者
C++入门教程:掌握函数重载、引用与内联函数的概念
通过上述介绍和实例,我们可以看到,函数重载提供了多态性;引用提高了函数调用的效率和便捷性;内联函数则在保证代码清晰的同时,提高了程序的运行效率。掌握这些概念,对于初学者来说是非常重要的,它们是提升C++编程技能的基石。
21 0
|
2月前
|
C++
C++(十五) 运算符重载
C++中的运算符重载允许对已有运算符的功能进行重新定义,从而扩展语言功能、简化代码并提升效率。重载遵循特定语法,如 `friend 类名 operator 运算符(参数)`。重载时需注意不可新增或改变运算符数量、语义、优先级、结合性和返回类型。常见示例包括双目运算符 `+=` 和单目运算符 `-` 及 `++`。输入输出流运算符 `&lt;&lt;` 和 `&gt;&gt;` 也可重载。部分运算符只能作为成员函数重载。
|
4月前
|
JSON Go C++
开发与运维C++问题之在iLogtail新架构中在C++主程序中新增插件的概念如何解决
开发与运维C++问题之在iLogtail新架构中在C++主程序中新增插件的概念如何解决
47 1
|
4月前
|
C++ 开发者
C++一分钟之-概念(concepts):C++20的类型约束
【7月更文挑战第4天】C++20引入了Concepts,提升模板编程的类型约束和可读性。概念定义了模板参数需遵循的规则。常见问题包括过度约束、约束不完整和重载决议复杂性。避免问题的关键在于适度约束、全面覆盖约束条件和理解重载决议。示例展示了如何用Concepts限制模板函数接受的类型。概念将增强模板的安全性和灵活性,但需谨慎使用以防止错误。随着C++的发展,Concepts将成为必备工具。
97 2
|
5月前
|
存储 编译器 C++
【C++】:拷贝构造函数和赋值运算符重载
【C++】:拷贝构造函数和赋值运算符重载
30 1
|
5月前
|
C++
C++一分钟之-继承与多态概念
【6月更文挑战第21天】**C++的继承与多态概述:** - 继承允许类从基类复用代码,增强代码结构和重用性。 - 多态通过虚函数实现,使不同类对象能以同一类型处理。 - 关键点包括访问权限、构造/析构、菱形问题、虚函数与动态绑定。 - 示例代码展示如何创建派生类和调用虚函数。 - 注意构造函数初始化、空指针检查和避免切片问题。 - 应用这些概念能提升程序设计和维护效率。
41 2
|
4月前
|
C++ 开发者
C++一分钟之-概念(concepts):C++20的类型约束
【7月更文挑战第6天】C++20引入了Concepts,提升模板编程的精确性和可读性。概念允许设定模板参数的编译时约束。常见问题包括过度约束、不完整约束及重载决议复杂性。要避免这些问题,需适度约束、全面覆盖约束条件并理解重载决议。示例展示了如何定义和使用`Incrementable`概念约束函数模板。概念是C++模板编程的强大力量,但也需谨慎使用以优化效率和代码质量。
105 0
|
4月前
|
自然语言处理 程序员 C++
C++基础知识(五:运算符重载)
运算符重载是C++中的一项强大特性,它允许程序员为自定义类型(如类或结构体)重新定义标准运算符的行为,使得这些运算符能够适用于自定义类型的操作。这样做可以增强代码的可读性和表达力,使得代码更接近自然语言,同时保持了面向对象编程的封装性。
|
4月前
|
Java 程序员 C++
|
4月前
|
编译器 C++
【C++】详解运算符重载,赋值运算符重载,++运算符重载
【C++】详解运算符重载,赋值运算符重载,++运算符重载