C++面向对象高级编程(六)转换函数与non-explicit one argument ctor

技术在于交流、沟通,转载请注明出处并保持作品的完整性。

1.conversion function 转换函数

    //1.转换函数
    //conversion function
    //只要你认为合理 你可以任意写转换函数
    class Fraction
    {
    public:
        Fraction(int num, int  den = 1):m_numerator(num),m_denominator(den){}

        operator double() const //注意看 没有返回类型,这个编译器会帮做,且防止我们声明错误 函数名就是返回类型
        {
            return ((double)m_numerator / m_denominator) ;//见下面自动转化规则
        }
    private:
        int m_numerator;
        int m_denominator;
    };

    int main(int argc, const char * argv[]) {

        Fraction f(3,5);
        double d = 4 + f;  //先找全局函数 operator+ , 发现没有,再去找  发现在 Fraction内找到了
        cout<< d << endl;
        cout<<sizeof(double) << endl;

        return 0;
    }

自动转换遵循以下规则:

1)若参与运算量的类型不同,则先转换成同一类型,然后进行运算。

2)转换按数据长度增加的方向进行,以保证精度不降低。如int型和long型运算时,先把int量转成long型后再进行运算。

a.若两种类型的字节数不同,转换成字节数高的类型

b.若两种类型的字节数相同,且一种有符号,一种无符号,则转换成无符号类型

3)所有的浮点运算都是以双精度进行的,即使仅含float单精度量运算的表达式,也要先转换成double型,再作运算。

4)char型和short型参与运算时,必须先转换成int型。

5)在赋值运算中,赋值号两边量的数据类型不同时,赋值号右边量的类型将转换为左边量的类型。如果右边量的数据类型长度左边长时,将丢失一部分数据,这样会降低精度,丢失的部分按四舍五入向前舍入。

例如:

上面的  return ((double)m_numerator / m_denominator) ;

如果声明成这样 (double) (m_numerator / m_denominator) 会执行int类型的除法运算

所以要先将两个int型参数的一个转换成double型参数即可



2.non-explicit one argument ctor

explicit的作用,防止隐式转换,一般作用于带参数的构造函数

class Fraction
{
public:
    Fraction(int num, int  den = 1):m_numerator(num),m_denominator(den){} //

    Fraction operator+(const Fraction & f)
    {
        cout << f.m_numerator<<endl;
        return f;
    }
private:
    int m_numerator;
    int m_denominator;
};

int main()
{
    Fraction f(3,5);
    Fraction d = f + 4;//调用时会使 4 隐式转换成 Fraction //调用的过程也是先去找f的operator+函数 找到了 调用
    return 0;
}

输出结果

Fraction d = f + 4; 这个函数的意义是 f调用operator+ 参数为4

而接受端

你会发现4被隐式转换成 Fraction

Fraction operator+(const Fraction & f){...}

参数4就被隐式转换成Fraction

那么加上 explicit

class Fraction
{
public:
    explicit Fraction(int num, int  den = 1):m_numerator(num),m_denominator(den){} //

    Fraction operator+(const Fraction & f)
    {
        cout << f.m_numerator<<endl;
        return f;
    }
private:
    int m_numerator;
    int m_denominator;
};

调用端



上面我提到两次 调用过程 那么如果将operator+  与 operator double() 连用的话会发生什么情况

class Fraction
{
public:
    /*explicit*/ Fraction(int num, int  den = 1):m_numerator(num),m_denominator(den){} //

    Fraction operator+(const Fraction & f)
    {
        cout << f.m_numerator<<endl;
        return f;
    }

    operator double() const //注意看 没有返回类型,这个编译器会帮做,且防止我们声明错误 函数名就是返回类型
    {
        return ((double)m_numerator / m_denominator) ;//见下面自动转化规则
    }
private:
    int m_numerator;
    int m_denominator;
};
    

编译会报错

参照<<侯捷 C++面向对象高级编程>>

时间: 2024-07-30 13:49:09

C++面向对象高级编程(六)转换函数与non-explicit one argument ctor的相关文章

C++面向对象高级编程(七)point-like classes和function-like classes

技术在于交流.沟通,转载请注明出处并保持作品的完整性. 1.pointer-like class 类设计成指针那样,可以当做指针来用,指针有两个常用操作符(*和->),所以我们必须重载这两个操作 /*简单实现一下智能指针的*与 -> 操作符*/ template <class T> class shared_ptr_test { public: T& operator* () const //重载* { return *px; } T* operator->() co

Python3 面向对象 高级编程

正常情况下,当我们定义了一个class,创建了一个class的实例后,我们可以给该实例绑定任何属性和方法,这就是动态语言的灵活性.  class Student(object): pass 然后,尝试给实例绑定一个属性: >>> s = Student() >>> s.name = 'Michael' # 动态给实例绑定一个属性 还可以尝试给实例绑定一个方法: >>> def set_age(self, age): # 定义一个函数作为实例方法 ...

C++面向对象高级编程(九)Reference与重载operator new和operator delete

摘要: 技术在于交流.沟通,转载请注明出处并保持作品的完整性. 一 Reference 引用:之前提及过,他的主要作用就是取别名,与指针很相似,实现也是基于指针. 1.引用必须有初值,且不能引用nullptr 2.引用之后不能再引用别人 3.引用通常不用于声明变量,多用于参数类型,和返回值类型 见下面代码 int main(int argc, const char * argv[]) { int x=0; //p is a pointer to x int* p = &x; // r is a

C++面向对象高级编程(三)

版权声明:本文为博主原创文章,未经博主允许不得转载. 接下来的几篇文章,我将回忆一下C++的基础. C++的由两部分组成 1.C++语言 2.C++标准库 本篇文章主要分享我学习C++语言的笔记. 本节主要介绍 Big Three 即析构函数,拷贝构造函数,赋值拷贝函数,前面主要围绕不带指针的class complex本节中主要围绕带指针的String类 前面我说过如果你创建的类不带有指针,那么多半你可以不用写析构函数,但是如果你创建了一个带指针的类,那么你必须重写Big Three 创建一个类

C++面向对象高级编程(二)

版权声明:本文为博主原创文章,未经博主允许不得转载. 接下来的几篇文章,我将回忆一下C++的基础. C++的由两部分组成 1.C++语言 2.C++标准库 本篇文章主要分享我学习C++语言的笔记. 这次主要回忆一下操作符重载. 先看一段代码,后续我会介绍这么做的原由 #include <iostream> class complex { public: complex(double r = 0, double i = 0) : re(r) ,im(i) {} complex& oper

[7]面向对象高级编程

[7]面向对象高级编程 数据封装.继承和多态只是面向对象程序设计中最基础的3个概念.在Python中,面向对象还有很多高级特性,允许我们写出非常强大的功能. 我们会讨论多重继承.定制类.元类等概念. 使用__slots__ 正常情况下,当我们定义了一个class,创建了一个class的实例后,我们可以给该实例绑定任何属性和方法,这就是动态语言的灵活性.先定义class: >>> class Student(object): ... pass ... 然后,尝试给实例绑定一个属性: >

Boolan C++面向对象高级编程学习第五周

本周C++面向对象高级编程编程主要是讲三个方面 1.虚表和虚指针 虚指针:如果类(包括父类)中有虚函数,就一定会有虚指针,虚指针是指向虚函数的虚表. 虚表:虚指针指向的一种类的表,表上存有类里面虚函数的实现的指针 这里虚函数和虚表一般是和类的多太性结合在一起来使用的,子类调用虚函数是通过调用虚指针来调用虚函数表里面的指针再来实现函数特性 这种调用的方式叫做动态绑定,是一种通过指针类型自动来判断调用的对象,在实现中可以不用区分哪个对象从而调用对应的函数 我们普通的调用类里面的函数叫做静态绑定 2.

python基础-面向对象高级编程

实例化对象绑定属性 s.name = 'Michael' # 动态给实例绑定一个属性 类绑定方法---所有实例化对象均可调用Student.set_score = set_score 实例化对象绑定方法---只有该对象可调用from types import MethodType#绑定方法需要使用MethodTypes.set_age = MethodType(set_age, s) # 给实例绑定一个方法 Python内置的@property装饰器就是负责把一个方法变成属性调用的把一个gett

UNIX环境高级编程之-----popen函数

头文件: #include <stdio.h> 函数说明: FILE * popen ( const char * command , const char * type ); int pclose ( FILE * stream ); 说明:(参考unix环境高级编程) popen() 函数通过创建一个管道,调用 fork 产生一个子进程,执行一个 shell 以运行命令来开启一个进程.这个进程必须由 pclose() 函数关闭,而不是 fclose() 函数.pclose() 函数关闭标准