详解C++中赋值,关系,函数调用运算符重载的实现

目录
  • 赋值运算符重载
    • 类结构
    • 问题的出现
    • 具体实现
  • 关系运算符重载
    • 类结构
    • 具体实现
  • 函数调用运算符重载
    • 类结构
    • 具体实现
  • 总结

赋值运算符重载

在C++中基本数据类型例如整型,可以实现连续赋值:a=b=c;而我们的对象的成员属性虽然可以相等,但是如果牵扯到堆地址,就会有深浅拷贝的问题存在。所以我们自己重载赋值运算符,实现连等的方法。

类结构

class Info
{
    int* m_a;
public:
    Info()
    {
        m_a = NULL;
    }
    Info(int a)
    {
        m_a = new int(a);
    }
    ~Info()
    {
        if (m_a != NULL)
            delete m_a;
        m_a = NULL;
    }

};

创建一个Info类,定义指针属性*m_a, 然后自己定义Info类的无参、有参构造函数和析构函数,再给属性赋值的时候,是用new关键字开辟堆区并存放数据;析构在前面的文章中讲到了他的作用,就是在程序结束前编译器自动调用析构来完成对象的清理工作,在这里就会把堆区的数据释放掉。

问题的出现

Info f(20);
Info f1;
Info f2;
Info f3(30);
f1 = f2 =f;

直接连等是不行的,因为在析构函数里访问了一个NULL地址,就比如程序即将结束,f调用析构函数,删除m_a指向的地址,然而f1也会调用析构函数,由于f1和f的m_a指向同一个地址,那就好重复访问,访问一个被删除的地址肯定会报错的;因此我们需要对赋值运算符进行重载,这里提示一下,连等就相当于链式调用,因此重载运算符的返回值类型需要返回引用。

具体实现

Info& operator=(Info& f)
    {
        if (m_a != NULL)
            delete m_a;
        m_a = NULL;
        m_a = new int(*f.m_a);
        return *this;
    }

返回值类型是类引用,这样可以做到链式调用 ,函数名还是统一的operator+运算符,既然是赋值运算符就用operator=,然后这个重载发生在成员内部,因此参数里只需要传入用来赋值的对象即可,注意倒数第二行代码,我利用new让m_a指向堆区中新开辟的地址,这是赋值运算符重载的关键;就是因为把地址指向了堆区的新地址,这样不同的对象在调用析构函数的时候各删各的堆地址,不会访问空地址,这个问题的解决和深浅拷贝的解决方式一样,都是自己写方法来避免原来方法中成员属性指向同一个地址。最后返回自身的引用,就可以实现连续调用了。

关系运算符重载

关系运算符有“大于”、“小于”、“等于”、“大于等于”、“不等于”等几种情况,我就举例等于和不等于两种赋值运算符重载的例子

类结构

class Info
{
    friend void test1();
    int* m_a;
    string m_name;
public:
    Info()
    {
        m_a = NULL;
    }
    Info(int a,string name)
    {
        m_a = new int(a);
        m_name = name;
    }
    ~Info()
    {
        if (m_a != NULL)
            delete m_a;
        m_a = NULL;
    }
}

这里的类结构相比于赋值运算符重载多了一个String类型的m_name属性,然后写出类的无参、有参构造和析构函数,最上面的friend关键字是加了一个友元的声明,让下面的test1函数可以访问类的私有属性。

具体实现

    bool operator==(Info& f)
    {
        if (*this->m_a == *f.m_a && m_name==f.m_name)
            return true;
        else return false;
    }
    bool operator!=(Info& f)
    {
        if (*this->m_a == *f.m_a && m_name == f.m_name)
            return false;
        else return true;
    }

返回值类型写成布尔类型,因为关系运算的结果就是布尔类型的,常和while循环以及if语句使用;函数名还是老样子,operator==和operator!=,分别是相等和不等;既然是成员内部的关系运算符重载,那么形参传入一个待比较对象即可。

调用方法

void test1()
{
    Info f1(20,"张三");
    Info f3(30,"张六");
    if (f1== f3) cout << "二者相等" << endl;
    else if (f1!= f3) cout << "二者不相等" << endl;
}

运行效果

函数调用运算符重载

函数调用使用“()”,函数调用也是可以重载的,而且重载的调用很像直接调用函数,因此也叫做仿函数。

类结构

class MyHello
{
public:
    string hello;
};

非常简单的类结构,只有一个公有权限下的String类型的hello属性。

具体实现

    void operator()(string s)
    {
        cout << s << endl;
    }

因为只是打印一下,不需要返回值,函数名不多说了,和前面类似,然后传入字符串类型再方法里打印出来。

调用方法

void test()
{
    MyHello hello;
    hello("Hello World");
}

首先创建类对象hello,直接使用重载后的调用方法:对象+(字符串);这样就能打印出引号里的内容了:

匿名对象调用

最后补充一个匿名对象的知识,示例:

class MyAdd
{
public:
    int operator()(int num1, int num2)
    {
        return num1 + num2;
    }
};
void test1()
{
    MyAdd myAdd;
    int num = myAdd(160, 40);
    cout << "ret =" << myAdd(160,40) << endl;
    cout << "ret =" << MyAdd()(100,50) << endl;
}

这里我写了一个只有重载函数调用函数的类,并在test1中用常规和匿名对象调用重载后的函数调用方法;看一下运行效果:

先创建对象,再通过对象调用函数的方法我们不感到奇怪,但是最后一个输出语句中,MyAdd()(100,50)是什么意思呢 ,这就是匿名对象,往后我们遇到形如类名+()再+调用函数的方式,那就是创建了匿名对象,其特点就是创建完毕后就会删除,这里我们只都一下数据,所以适合匿名对象的调用。

总结

到这里C++的运算符重载彻底结束了,运算符重载可以多个都重载并互相配合使用,把对象的属性可以和基本类型一样进行算数运算和逻辑运算,在类对象比较多的时候可以节省很多时间,简化程序。

以上就是详解C++中赋值,关系,函数调用运算符重载的实现的详细内容,更多关于C++运算符重载的资料请关注我们其它相关文章!

(0)

相关推荐

  • C++编程语言中赋值运算符重载函数(operator=)的使用

    目录 1 概述 1.1 Why 2 示例代码 2.1 示例代码1 2.2 示例代码2 3 总结 本文主要介绍 C++ 编程语言中赋值运算符重载函数(operator=)的相关知识,同时通过示例代码介绍赋值运算符重载函数的使用方法. 1 概述 1.1 Why 首先介绍为什么要对赋值运算符“=”进行重载.某些情况下,当我们编写一个类的时候,并不需要为该类重载“=”运算符,因为编译系统为每个类提供了默认的赋值运算符“=”,使用这个默认的赋值运算符操作类对象时,该运算符会把这个类的所有数据成员都进行一次

  • C++简单又好用的基本运算符重载

    目录 运算符重载概念 加号运算符重载 成员函数实现 全局函数实现 运算符实现函数重载 总结 运算符重载概念 对已有的运算符进行重新定义,赋予其另外一种功能,以适应不同的数据类型 我们知道已有的运算符有'+'.'-'.'*'.'/'等,这些运算符对于内置数据类型可以直接使用,例如int.float.double.char等等.但是如果我们定义一个类,想实现类中对象属性的加减乘除运算,该怎么实现呢?那就用到运算符重载的知识点了. 加号运算符重载 学会一个顶四个,这篇博文只举例加号运算符重载 成员函数

  • 详解C++中的函数调用和下标以及成员访问运算符的重载

    函数调用 使用括号调用的函数调用运算符是二元运算符. 语法 primary-expression ( expression-list ) 备注 在此上下文中,primary-expression 为第一个操作数,并且 expression-list(可能为参数的空列表)为第二个操作数.函数调用运算符用于需要大量参数的操作.这之所以有效,是因为 expression-list 是列表而非单一操作数.函数调用运算符必须是非静态成员函数. 函数调用运算符在重载时不会修改函数的调用方式:相反,它会在运算

  • C++ 类的赋值运算符''=''重载的方法实现

    什么类需要重载赋值运算符 先来看一个普通类的直接赋值. #include <iostream> using namespace std; class person{ int age; public: person(const int& a=10):age(a){} //构造函数 ~person(); //析构函数 void showAdd(); //打印age的地址 }; person::~person(){cout<<"析构\n";} void per

  • 详解C++中递增运算符重载的实现

    目录 递增运算符 递增运算符重载的实现 左移运算符重载 前置递增运算符重载 后置递增运算符重载 递增运算符 形如"a++"."++a".如果a初始值为1,那么"a++"结果为1,但是实际上a等于2,先读取再加1:"++a"结果为2,实际也是2.这是对于基本运算类型,那么递增运算符重载的目的就是对于对象的属性也可以直接进行前置递增和后置递增. 由于成员函数里重载写的内容少,就详细分析成员函数实现递增运算符重载 递增运算符重载的实

  • C++超详细讲解运算符重载

    目录 概念 赋值运算符重载 const成员 取地址及const取地址操作符重载 概念 C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类 型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似. 函数名字为:关键字operator后面接需要重载的运算符符号. 函数原型:返回值类型 operator操作符(参数列表) 需要注意的几点: 不能通过连接其他符号来创建新的操作符:比如operator@,必须是已有的操作符: 重载操作符必须有一个类类型

  • 详解C++中赋值,关系,函数调用运算符重载的实现

    目录 赋值运算符重载 类结构 问题的出现 具体实现 关系运算符重载 类结构 具体实现 函数调用运算符重载 类结构 具体实现 总结 赋值运算符重载 在C++中基本数据类型例如整型,可以实现连续赋值:a=b=c:而我们的对象的成员属性虽然可以相等,但是如果牵扯到堆地址,就会有深浅拷贝的问题存在.所以我们自己重载赋值运算符,实现连等的方法. 类结构 class Info { int* m_a; public: Info() { m_a = NULL; } Info(int a) { m_a = new

  • 详解java中继承关系类加载顺序问题

    详解java中继承关系类加载顺序问题 实例代码: /** * Created by fei on 2017/5/31. */ public class SonClass extends ParentClass{ public SonClass(){ System.out.println("SonClass's constructor"); } { System.out.println("SonClass's block");} static { System.out

  • 详解Java中方法重写和方法重载的6个区别

    目录 1.方法重写 1.1 基本用法 1.2 使用场景 1.3 注意事项 2.方法重载 2.1 基本使用 2.2 使用场景 2.3 注意事项 3.方法重写 VS 方法重载 总结 方法重写(Override)和方法重载(Overload)都是面向对象编程中,多态特性的不同体现,但二者本身并无关联,它们的区别犹如马德华之于刘德华的区别,除了名字长得像之外,其他的都不像. 接下来咱们就来扒一下二者的具体区别. 1.方法重写 方法重写(Override)是一种语言特性,它是多态的具体表现,它允许子类重新

  • 详解C++中赋值和输入输出语句的用法

    C++赋值语句讲解 C++的赋值语句具有其他高级语言的赋值语句的功能.但不同的是,C++中的赋值号"="是一个运算符,可以写成 a=b=c=d; 而在其他大多数语言中赋值号不是运算符,上面的写法是不合法的. 关于赋值表达式与赋值语句的概念.在C++中,赋值表达式可以包括在其他表达式之中,例如: if((a=b)>0) cout<<"a>0"<<endl; 按语法规定if后面的( )内是一个条件.现在在x的位置上换上一个赋值表达式&

  • 详解C++中的成员访问运算符和指针到成员运算符

    成员访问运算符:. 和 -> 语法 postfix-expression       . name postfix-expression –> name 备注 成员访问运算符 . 和 -> 用来引用结构.联合和类的成员.成员访问表达式具有选定成员的值和类型. 有两种形式的成员访问表达式: 在第一种形式中,postfix-expression 表示结构.类或联合类型的值,name 为指定的结构.联合或类的成员命名.运算的值是 name 的值且为左值(如果 postfix-expressio

  • 详解PHP中的null合并运算符

    null合并运算符是一个好东西,有了它我们就能很方便的获取一个参数,并能在其为空的情况下提供一个默认值.比如在js中可以用||来搞: function setSomething(a){ a = a || 'some-default-value'; // ... } 而在PHP中,可惜PHP的||总是返回true或false,无法这样来搞.  PHP7才正式加入了??这个运算符: // 获取user参数的值(如果为空,则用'nobody') $username = $_GET['user'] ??

  • 详解JavaScript中|单竖杠运算符的使用方法

    js运算符单竖杠"|"的作用 在js整数操作的时候,相当于去除小数点,parseInt.在正数的时候相当于Math.floor(),负数的时候相当于Math.ceil() 注: 1. Math.ceil()用作向上取整. 2. Math.floor()用作向下取整. 3. Math.round() 我们数学中常用到的四舍五入取整. console.log(0.6|0)//0 console.log(1.1|0)//1 console.log(3.65555|0)//3 console.

  • 详解C++中const_cast与reinterpret_cast运算符的用法

    const_cast 运算符 从类中移除 const.volatile 和 __unaligned 特性. 语法 const_cast < type-id > ( expression ) 备注 指向任何对象类型的指针或指向数据成员的指针可显式转换为完全相同的类型(const.volatile 和 __unaligned 限定符除外).对于指针和引用,结果将引用原始对象.对于指向数据成员的指针,结果将引用与指向数据成员的原始(未强制转换)的指针相同的成员.根据引用对象的类型,通过生成的指针.引

  • C 语言基础----详解C中的运算符

    C语言中又有哪些运算符呢? 如下所示: ※ 算术运算符 ※ 赋值运算符 ※ 关系运算符 ※ 逻辑运算符 ※ 三目运算符 C语言基本算术运算符如下表: 除法运算中注意: 如果相除的两个数都是整数的话,则结果也为整数,小数部分省略,如果两数中有一个为小数,结果则为小数. 取余运算中注意: 该运算只适合用两个整数进行取余运算 运算后的符号取决于被模数的符号,如(-10)%3 = -1;而10%(-3) = 1. 注:C语言中没有乘方这个运算符,也不能用×,÷等算术符号. 赋值运算符 下表列出了 C 语

  • 一文详解C++中运算符的使用

    目录 一.算术运算符 二.关系运算符 三.逻辑运算符 四.位运算符 五.赋值运算符 六.杂项运算符 一.算术运算符 运算符 描述 + 把两个操作数相加 - 从第一个操作数中减去第二个操作数 * 把两个操作数相乘 / 分子除以分母 % 取模运算符,整除后的余数 ++ 自增运算符,整数值增加 1 – 自减运算符,整数值减少 1 通过下面的例子可以让我们更好的理解C++中的运算符的意义与使用方法. #include <iostream> using namespace std; int main()

随机推荐