C++—使用类

运算符重载

概念

  • 运算符重载就是想法转换, 目的是简化函数调用的方式
  • 重载就是赋予新的含义, 运算符重载也是, 即同一个运算符可以有不同的功能
  • C++本身已经对一些运算符进行了重载, 同时C++允许程序员自己重载运算符
    • +号可以对不同类型(int float)的数据进行加法操作
    • << 既是移位运算符, 又可以配合cout向控制台输出数据
  • 定义一个运算符重载就像定义一个函数一样, 只不过这个函数名称以operator关键字开头
    • 返回类型 operator被重载的运算符(参数列表)

可以重载的运算符

运算符
关系运算符 == != < > <= >=
逻辑运算符 || && !
一元运算符 + - * & ++ --
位运算符 | & ~ ^ << >>
赋值运算符 = += -= *= /= %= &= |= ^= <<= >>=
内存声明与释放 new delete new[] delete[]
其他运算符 函数调用() 成员访问 -> 成员指针访问 ->* 逗号 , 下标 []

不能重载的运算

  • 点运算符 .
  • 成员指针访问运算符 *
  • 域运算符 ::
  • 长度运算符 sizeof
  • 三元运算符 ? :

注意

  • 重载不能修改运算变量的个数, 比如, 关系运算是二元运算, 重载后也必须有两个变量参数运算
  • 重载不能修改运算符的优先级别, 比如, 乘除优先于加减, 重载后这个优先级不会被修改
  • 重载不能修改运算顺序, 比如, 赋值运算是从右到左的, 重载后不能改变
class Integer
{
    public:
        //构造函数
        Integer(): m_num(0){}
        Integer(int num): m_num(num){}

        const Integer operator+(const Integer & other) const 
        {
            cout << "重载了+号运算符, 以实现两个Integer的对象的相加" << endl;
            return Integer(this->m_num + other.m_num);
        }
    private:
        int m_num;
}

Integer num1(1024);
Integer num2(20);
Integer num3 = num1 + num2;   //编译器实际调用 num3 = num1.operator+(num2);

const的作用

  • const修饰成员变量

    • 只有一个const时, 如果const位于*的左侧, 表示指针所指的数据是常量, 不能通过该指针修改实际数据

      const int num = 1024;
      num = 2048;             //不合法
      
      //const在*号左侧, 表示指针所指向数据是常量
      const int * ptr1_num1 = &num1;
      int const * ptr2_num1 = &num;
      prt1_num = &num;        //合法, 可以指向其他内存单元
      *ptr_num1 = 1234;       //不合法, 不能修改所指向的数据
      
    • 只有一个const时, 如果const位于*的右侧, 表示指针本身是常量, 不能指向其他内存单元, 所指向的数据可以修改

      //const在*右侧, 表示指针本身是常量, 不能指向其他内存单元
      int * const ptr3_num1 = &num1;
      ptr3_num1 = ptr2_num;
      
    • 如果有两个const位于*的左右两侧, 表示指针和指针所指向的数据都不能修改

  • const修饰函数参数

    • const修饰引用时, 不能修改引用对象的任何成员, 可以保护传递的参数, 起到不copy对象的目的, 节省效率
  • const修饰返回值

    • 如果函数要返回局部对象, 直接返回这个对象即可, 不要返回这个对象的引用
    • 在可以返回对象, 也可以返回对象的引用时, 首选返回引用, 提高效率
  • const修饰函数,时, 说明函数不会修改成员变量的值

友元函数

  • 对于很多运算符来说, 可以选择使用成员函数或非成员函数来实现运算符重载, 通常非成员函数就是友元函数, 这样可以直接访问类的私有数据
  • 在定义运算符时, 不能同时选择这两种格式, 同时定义两种格式将被视为二义性错误, 出现编译错误

建议准则:

  • C++规定, 赋值运算符=, 数组下标运算符[], 函数调用运算符(), 成员访问运算符->, 在重载时必须声明为类的成员函数

  • 一元运算符和复合赋值运算符重载时, 一般声明为类的成员函数

  • 流运算符<< >>, 类型转换运算符不能定义为类的成员函数, 只能是友元函数

  • 二元运算符在运算符重载时, 一般声明为友元函数

firend ostream& operator<<(ostream& out, const Hero& hero);

firend ostream& operator<<(ostream& out, const Hero& hero)
{
    out << "昵称" << hero.GetNickName() << '\n';
    out << "等级" << hero.GetLevel() << '\n';
    return out;
}

赋值/拷贝构造函数

  • 简单的类, 默认拷贝构造函数一般就构用, 不需要显式地定义一个功能类似的拷贝构造函数

  • 当类拥有其他资源时, 如动态分配的内存, 打开的文件, 指向其他数据的指针, 网络连接等, 默认拷贝构造函数就不能拷贝这些资源, 必须显式地定义拷贝构造函数, 以完整地拷贝对象的所有数据

  • 为类定义赋值/拷贝构造函数, 即以对象为参数的构造函数

    Student::Student(Student & stu_ptr)
    
    //const关键字可以保证复制过程中不会改变被复制的对象
    Student::Student(const Student & stu_ptr)
    
  • 下面三张场景会调用复制构造函数

    • 当类的对象被初始化为同一类的另一个对象时
    • 当对象被作为参数传递给一个函数时
    • 当函数返回一个对象时

类型转换

  • C++中, 存在隐式类型转换语法, 即自动类型转换

    int a = 12;
    a = 22.9 + a;
    
  • C++还提供了显式类型转换语法, 即强制类型转换

    int num = int(89.3);
    
  • C语言中,采用以下语法

    int num = (int)89.3;
    
  • 注意

    • 将浮点型数据赋值给整型变量时, 会舍弃小数部分
    • 将整型数据赋值给浮点型变量时, 数值不变, 但是会以指数形式存储
    • 将double性数据赋值给float型变量时, 注意数值范围溢出
    • 字符型数据可以赋值给整型变量, 存入的是字符的ASCII码
    • 将一个int, short或long型赋值给char类型变量时, 只将低8位传给char变量
    • 将有符号型数据赋值给长度相同的无符号型变量时, 连同原来的符号位一起传送
  • C++允许自定义类型之间转换, 但是自定义的类型转换规则只能以类的成员函数的形式出现

  • 将其他类型转换为当前类的类型时, 需要借助转换构造函数

再谈构造函数

  • 构造函数是在创建对象时, 初始化对象
  • 编译会根据传递的实参来匹配不同的构造函数
//无参构造
Rectangle();

Rectangle rect1();

//带参构造
Rectangle(float width, float height);

Rectangle rect2(20, 30);

//拷贝构造, 以拷贝的方式初始化对象时调用
Rectangle(const Rectangle& rect);

Rectangle rect3(rect2);

//转换构造, 将其他类型转换为当前类型时调用
Rectangle(float width);

Rectangle rect4(99.8);
Rectangle rect5 = 66.6;
Rectangle rect6;
rect6 = rect5 + 'A' + false;   //将char, bool都转换为Rectangle类型再运算
  • 当前类的类型转换为其他类型时, 使用类型转换函数

  • 类型转换函数只能以成员函数的形式出现, 也就是只能出现在类中

    //类型转换函数的语法格式
    operator type()
    {
        return data;
    }
    
    /*
    type可以是内置类型, 类类型以及由typedef定义的类型别名, 任何作为函数返回类型的类型都是被支持的, void除外, 且不允许转换为数组或函数类型, 可以转换为指针或引用类型
    */
    
  • 类型转换函数看起来没有返回值类型, 但其实隐式地指明了返回值类型

  • 类型转换函数也没有参数, 因为要将当前类的对象转换为其他类型, 编译器实际上会把当前对象的地址赋值给this指针, 这样就可以在函数体内可以操作当前对象

    operator float() const{
        return this->width;
    }
    
    //类型转换函数一般不会更改被转换的对象, 所以通常被定义为const
    //类型转换函数可以被继承, 可以是虚函数
    

原文链接: https://www.cnblogs.com/KX-Lau/p/12857601.html

欢迎关注

微信关注下方公众号,第一时间获取干货硬货;公众号内回复【pdf】免费获取数百本计算机经典书籍;

也有高质量的技术群,里面有嵌入式、搜广推等BAT大佬

    C++---使用类

原创文章受到原创版权保护。转载请注明出处:https://www.ccppcoding.com/archives/347292

非原创文章文中已经注明原地址,如有侵权,联系删除

关注公众号【高性能架构探索】,第一时间获取最新文章

转载文章受原作者版权保护。转载请注明原作者出处!

(0)
上一篇 2023年3月2日 上午4:26
下一篇 2023年3月2日 上午4:27

相关推荐