『C++成长记』运算符重载

这篇具有很好参考价值的文章主要介绍了『C++成长记』运算符重载。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

『C++成长记』运算符重载,C++,c++,开发语言

🔥博客主页:小王又困了

📚系列专栏:C++

🌟人之为学,不日近则日退

❤️感谢大家点赞👍收藏⭐评论✍️

『C++成长记』运算符重载,C++,c++,开发语言

目录

一、运算符重载

📒1.1两个日期大小的比较

📒1.2运算符重载的引入

📒1.3将运算符重载函数写成成员函数

二、赋值运算符重载


一、运算符重载

int main()
{
    int x = 1, y = 2;
    bool x > y;

    Date d1;
    Date d2(2023, 11, 20);
    bool d1 > d2;//不支持
    return 0;
}

     我们比较两个对象的大小,对于内置类型,可以直接使用运算符,内置类型都是简单类型,语言自己定义的,编译器直接转换成指令;对于自定义类型,编译器不知道它的行为,所以不支持直接使用运算符。那自定义类型该如何比较大小呢?我们以日期类为例。

📒1.1两个日期大小的比较

class Date
{
public:
    Date(int year, int month, int day)
    {
        _year = year;
        _month = month;
        _day = day;
    }
private:
    int _year;//年
    int _month;//月
    int _day;//日
};

int main()
{
	Date d1(2023, 5, 21);
	Date d2(2023, 11, 20);
	return 0;
}

定义了d1d2两个对象,我们可以通过函数比较两个日期的大小,如下:

//以大于为例
bool Greater(const Date& x, const Date& y)
{
    if (x._year > y._year)
    {
        return true;
    }
    else if (x._year == y._year && x._month > y._month)
    {
        return true;
    }
    else if (x._year == y._year && x._month == y._month && x._day > y._day)
    {
        return true;
    }
    
    return false;
}

int main()
{
    Date d1(2023, 5, 21);
    Date d2(2023, 11, 20);
    cout << Greater(d1, d2) << endl;
    return 0;
}

但这里有一个问题,Greater函数是定义在类外的,日期类的成员变量是私有的,所以在类外是无法访问到的,想要实现比较的功能就要将成员变量设为public公有,这样代码是不安全的。

📒1.2运算符重载的引入

     有些人给函数命名时并不规范,例如:Compare1Compare2等。我们不知道这个函数实现的是什么功能,只有看过代码才知道,这样十分麻烦。C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。

  • 函数名字:关键字operator后面接需要重载的运算符符号
  • 函数原型:返回值类型 operator操作符(参数列表)
//以大于为例
bool operator>(const Date& x, const Date& y)
{
    if (x._year > y._year)
    {
        return true;
    }
    else if (x._year == y._year && x._month > y._month)
    {
        return true;
    }
    else if (x._year == y._year && x._month == y._month && x._day > y._day)
    {
        return true;
    }
    
    return false;
}

int main()
{
    Date d1(2023, 5, 21);
    Date d2(2023, 11, 20);
    cout << (d1 > d2) << endl;
    return 0;
}

注意:流插入的优先级高,我们要先比较大小,所以d1>d2要加括号。

📒1.3将运算符重载函数写成成员函数

     为了解决上面的私有成员变量在类外面无法访问的问题,可以把运算符重载函数写成类的成员函数或者友元,这样就能访问到私有的成员变量,但是友元一般不建议使用,因为友元会破坏封装。

class Date
{
public:
    Date(int year, int month, int day)
    {
        _year = year;
        _month = month;
        _day = day;
    }

    bool operator>(const Date& y)
    {
        if (year > y._year)
        {
            return true;
        }
        else if (year == y._year && month > y._month)
        {
            return true;
        }
        else if (year == y._year && month == y._month && day > y._day)
        {
            return true;
        }

        return false;
    }

private:
    int _year;//年
    int _month;//月
    int _day;//日
};

运算符重载有一个规定,运算符要与参数个数匹配, 是双目操作符,但上面的代码为什么只传递了一个参数,这是因为成员函数中都有一个隐藏的this,所以形参就只需要一个。

int main()
{
    d1 > d2;
    d1.operator>(&d1, d2);//显示调用
    return 0;
}

d1 > d2会转化为 d1.operator>(&d1, d2)

运算符重载有意义就可以实现,没有意义就不要实现。例如:日期+日期就没有意义,但日期-日期就有意义。

📝注意:

  • 不能通过连接其他符号来创建新的操作符:比如operator@。
  • 重载操作符必须有一个类类型参数。
  • 用于内置类型的运算符,其含义不能改变,例如:内置的+,不能改变其含义。
  • 作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐藏的this
  •  .*  :: sizeof  ?: 注意以上5个运算符不能重载。这个经常在笔试选择题中出现。

二、赋值运算符重载

📝区分赋值运算符重载和拷贝构造 

Date d1(2020, 5, 21);
Date d2(2023, 6, 21);
d1 = d2;//需要调用赋值运算符重载
Date d3(d1);//调用拷贝构造

要区分赋值运算符重载拷贝构造,前者是针对两个已存在的对象,将一个对象的值,赋值给另一个,而后者是用一个已存在的对象去初始化创建一个新对象。

赋值运算符重载格式:

  • 参数类型const T&(T是类型),传递引用可以提高传参效率
  • 返回值类型T&,返回引用可以提高返回的效率,有返回值目的是为了支持连续赋值
  • 检测是否自己给自己赋值
  • 返回*this :要复合连续赋值的含义
Date& operator=(const Data& d)
{
    if (this != &d)
    {
        _year = d._year;
        _month = d._month;
        _day = d._day;
	}
    return *this;//出了作用域*this还在,所以可以用引用返回
}

📝赋值运算符只能重载成类的成员函数不能重载成全局函数 

// 赋值运算符重载成全局函数,注意重载成全局函数时没有this指针了,需要给两个参数
Date& operator=(Date& left, const Date& right)
{
    if (&left != &right)
    {
        left._year = right._year;
        left._month = right._month;
        left._day = right._day;
    }
    return left;
}

原因:赋值运算符如果不显式实现,编译器会生成一个默认的。此时用户再在类外自己实现 一个全局的赋值运算符重载,就和编译器在类中生成的默认赋值运算符重载冲突了,故赋值运算符重载只能是类的成员函数。

📝编译器生成的默认赋值运算符重载做了什么

     用户没有显式实现时,编译器生成的默认赋值运算符重载,对内置类型的成员变量是以值的方式逐字节进行拷贝(浅拷贝),对自定义类型的成员变量,调用其对应类的赋值运算符重载。


🎁结语: 

     本次的内容到这里就结束啦。希望大家阅读完可以有所收获,同时也感谢各位读者三连支持。文章有问题可以在评论区留言,博主一定认真认真修改,以后写出更好的文章。你们的支持就是博主最大的动力。文章来源地址https://www.toymoban.com/news/detail-775969.html

到了这里,关于『C++成长记』运算符重载的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【C++】运算符重载

    目录 1. 基本概念 1.1 直接调用一个重载的运算符函数 1.2 某些运算符不应该被重载 1.3 使用与内置类型一致的含义 1.4 赋值和复合赋值运算符 1.5 选择作为成员或者非成员 2. 输入和输出运算符 2.1 输出运算符重载 2.2 输入运算符重载 3. 算术和关系运算符 3.1 算数运算符重载 3.2

    2024年02月11日
    浏览(36)
  • C++:重载运算符

    1.重载不能改变运算符运算的对象个数 2.重载不能改变运算符的优先级别 3.重载不能改变运算符的结合性 4.重载运算符必须和用户定义的自定义类型的对象一起使用,其参数至少应该有一个是类对象,或类对象的引用 5.重载运算符的功能要类似于该运算符作用于标准类型数据

    2024年02月10日
    浏览(37)
  • 复习 --- C++运算符重载

    .5 运算符重载 运算符重载概念:对已有的运算符重新进行定义,赋予其另外一种功能,以适应不同的数据类型 4.5.1 加号运算符重载 作用:实现两个自定义数据类型相加的运算 4.5.2 左移运算符重载 4.5.3递增运算符重载 作用:通过重载递增运算符,实现自己的整型数据 4.5.4 赋

    2024年02月07日
    浏览(35)
  • C++——类和对象3|日期类型|Cout运算符重载|Cin运算符重载|const成员|

    目录 日期类型  Date.h  Date.cpp  Test.cpp  实现Cout运算符重载  实现Cin运算符重载  根据日期算星期  修改后完整代码   Date.h  Date.cpp  const成员  取地址及const取地址操作符重载 习题  计算日期到天数转换     一个类到底可以重载哪些运算符,要看哪些运算符对这个类型有

    2023年04月13日
    浏览(49)
  • C++中的重载运算符

    🐶博主主页: @ᰔᩚ. 一怀明月ꦿ   ❤️‍🔥 专栏系列: 线性代数,C初学者入门训练,题解C,C的使用文章,「初学」C++ 🔥 座右铭: “不要等到什么都没有了,才下定决心去做” 🚀🚀🚀大家觉不错的话,就恳求大家点点关注,点点小爱心,指点指点🚀🚀🚀 目录 🐰

    2024年02月05日
    浏览(41)
  • C++中重载相等运算符 ==

    相等运算(==)是一种关系运算,与不等运算(!=)关系密切。 通常情况下,C++ 中的类通过定义相等运算符来检验两个对象是否相等。也就是说它们会比较对象的每一个数据成员,只有当所有对应的 成员都相等时才认为两个对象相等。依据这一思想,我们的 Sales_data 类的相等

    2024年02月12日
    浏览(32)
  • C++中重载输出运算符 <<

    C++输入输出标准库提供了“”和“”运算符执行输入、输出操作,但标准库只定义了基本数据类型的输入、输出操作,若要直接对类对象进行输入、输出,则需要在类中重载这两个运算符。与其他运算符不同的是, 输入、输出运算符只能重载成类的友元函数 。 通常情况下,

    2024年02月13日
    浏览(55)
  • C++语法——详解运算符重载

    运算符重载是C++的一个重要特性。有了运算符重载,在代码编写时能更好的实现封装。 目录 一.运算符重载介绍 二.运算符重载形式 (一).参数 (二).返回值 (三).应用 三.特殊的运算符重载 (一).默认赋值运算符重载 (二).自增运算符A++与++A (三).流提取与流插入

    2023年04月25日
    浏览(42)
  • c++ 友元 运算符重载详解

    c++是面向对象的,目的之一:封装 封装: 优点之一,就是安全。 缺点:在某些特殊的场合,不是很方便。 华为与IBM 40亿的咨询故事 IBM需要对华为各级部门做深度咨询分析, 为了提高咨询效率,由任正非直接授权,直接获取各部门的所有权限。 使用前提: 某个类需要实现

    2024年02月11日
    浏览(30)
  • C++ 面向对象(3)——重载运算符和重载函数

    C++ 允许在同一作用域中的某个 函数 和 运算符 指定多个定义,分别称为 函数重载 和 运算符重载 。 重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它们的参数列表和定义(实现)不相同。 当您调用一个 重载函数 或 重载运算符 时

    2024年02月10日
    浏览(39)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包