第五层:C++中的运算符重载(下)

简介: 第五层:C++中的运算符重载(下)

后置

当定义后置++的时候,发现,与前置++使用的是一样的名字,发现了重定义,那这个时候怎么办?可以在参数中写一个int,用于区分,这个int属于占位参数,用于区分前置递增和后置递增,编译器是只认int的,对于后置递增,应该记录初始结果,如果在递增,返回初始结果,这个时候应该返回值,而不是引用,因为返回的是一个局部变量,如果返回引用,就会非法访问:


不是用引用

#include<iostream>
using namespace std;
class A
{
public:
  A(int a)
  {
  _a = a;
  }
  //后置++
  A operator++(int)
  {
  int a = this->_a++;
  return a;
  }
  int _a;
};
void test1()
{
  A a1(10);
  cout << a1._a++ << endl;
  cout << a1._a << endl;
}
int main()
{
  test1();
  return 0;
}

0a2653c851af460fa595bd959398a8f1.png

是引用

#include<iostream>
using namespace std;
class A
{
public:
  A(int a)
  {
  _a = a;
  }
  //后置++
  A& operator++(int)
  {
  int a = this->_a++;
  return a;
  }
  int _a;
};
void test1()
{
  A a1(10);
  cout << a1._a++ << endl;
  cout << a1._a << endl;
}
int main()
{
  test1();
  return 0;
}

0eacb84100b54626af849e6b562bf92a.png


赋值运算符重载


其实在C++中,编译器一般会给一个类默认提供四个函数

1.默认构造函数(无参,函数体为空)

2.默认析构函数(无参,函数体为空)

3.默认拷贝函数,对属性进行浅拷贝

4.赋值运算符,对属性进行浅拷贝

注意!:


当这个时候属性内的值有属性指向堆区时,赋值操作符也会产生深拷贝和浅拷贝问题

编译器提供的是浅拷贝,这个时候在析构函数内进行delete释放内存时,会出现内存释放两次,所以这个时候operator=修改成深拷贝即可,注意,在赋值操作符之前,应该先判断赋值左侧值内堆区是否有数据,有数据要先释放干净,在进行深拷贝,记得返回值要返回自身:


#include<iostream>
using namespace std;
class A
{
public:
  A()
  {
  p = new int;
  }
  ~A()
  {
  delete p;
  p = NULL;
  }
  void operator=(A& a)
  {
  if (p != NULL)
  {
    p = NULL;
  }
  p = new int(*a.p);
  }
  int* p;
};
void test1()
{
  A a;
  *a.p = 10;
  A b;
  b = a;
  cout << *b.p << endl;
}
int main()
{
  test1();
  return 0;
}

0a2653c851af460fa595bd959398a8f1.png


关系运算符重载


作用


可以人两个自定义类型进行比较操作


实现


只用对比每一组属性是否相等,相等返回真,不相等返回假:


#include<iostream>
using namespace std;
class A
{
public:
  A(int a, char c)
  {
  _a = a;
  _c = c;
  }
  bool operator==(A& a)
  {
  if (_a == a._a && _c == a._c)
  {
    return 1;
  }
  return 0;
  }
  int _a;
  char _c;
};
void test1()
{
  A a(10, 'c');
  A b(10, 'c');
  if (a == b)
  {
  cout << "a=b" << endl;
  }
  else
  {
  cout << "a!=b" << endl;
  }
}
int main()
{
  test1();
  return 0;
}

0eacb84100b54626af849e6b562bf92a.png


函数调用运算符重载


函数调用操作符()也可以重载

由于重载后调用的方式和函数本身的调用相似,所以也被称为仿函数

仿函数没有固定写法,函数体内可以写任何。

事例:


#include<iostream>
using namespace std;
class A
{
public:
  void operator()()
  {
  cout << "仿函数调用" << endl;
  }
};
void test1()
{
  A a;
  a();
}
int main()
{
  test1();
  return 0;
}

2d65d23f6d4748949b924e4057485923.png


第二种重载掌握!突破


面前的石碑碎去,露出后面黑黝黝的洞口…


本章知识点(图片形式)


2e9b90b2ca334476abebe75bafe6eeaa.png


相关文章
|
6月前
|
编译器 C++
C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)
C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)
53 1
|
7月前
|
程序员 编译器 C++
C++中的运算符重载(Operator Overloading)
C++中的运算符重载(Operator Overloading)
60 1
|
3月前
|
C++
C++(十五) 运算符重载
C++中的运算符重载允许对已有运算符的功能进行重新定义,从而扩展语言功能、简化代码并提升效率。重载遵循特定语法,如 `friend 类名 operator 运算符(参数)`。重载时需注意不可新增或改变运算符数量、语义、优先级、结合性和返回类型。常见示例包括双目运算符 `+=` 和单目运算符 `-` 及 `++`。输入输出流运算符 `&lt;&lt;` 和 `&gt;&gt;` 也可重载。部分运算符只能作为成员函数重载。
|
6月前
|
存储 编译器 C++
【C++】:拷贝构造函数和赋值运算符重载
【C++】:拷贝构造函数和赋值运算符重载
30 1
|
6月前
|
C++ 索引
C++核心技术要点《运算符重载》
C++核心技术要点《运算符重载》
54 2
|
5月前
|
自然语言处理 程序员 C++
C++基础知识(五:运算符重载)
运算符重载是C++中的一项强大特性,它允许程序员为自定义类型(如类或结构体)重新定义标准运算符的行为,使得这些运算符能够适用于自定义类型的操作。这样做可以增强代码的可读性和表达力,使得代码更接近自然语言,同时保持了面向对象编程的封装性。
|
5月前
|
Java 程序员 C++
|
5月前
|
编译器 C++
【C++】详解运算符重载,赋值运算符重载,++运算符重载
【C++】详解运算符重载,赋值运算符重载,++运算符重载
|
6月前
|
编译器 C++
【C++】类和对象③(类的默认成员函数:赋值运算符重载)
在C++中,运算符重载允许为用户定义的类型扩展运算符功能,但不能创建新运算符如`operator@`。重载的运算符必须至少有一个类类型参数,且不能改变内置类型运算符的含义。`.*::sizeof?`不可重载。赋值运算符`=`通常作为成员函数重载,确保封装性,如`Date`类的`operator==`。赋值运算符应返回引用并检查自我赋值。当未显式重载时,编译器提供默认实现,但这可能不足以处理资源管理。拷贝构造和赋值运算符在对象复制中有不同用途,需根据类需求定制实现。正确实现它们对避免数据错误和内存问题至关重要。接下来将探讨更多操作符重载和默认成员函数。
|
6月前
|
C++
c++进阶篇(一)——运算符重载
c++进阶篇(一)——运算符重载