转换函数(conversion function),类中基本类型转换函数。1.转换函数必须是类方法; 2.转换函数不能指定返回类型; 3.转换函数不能有参数。
1.代码
设计一个类Fraction表示分数,包含分子和分母。我们希望它能自动转换为double类型,并参与运算。
Fraction类的代码如下,下文会对该代码做详细的分析。
class Fraction
{
public:
Fraction(int numerator, int denominator = 1)
:m_numerator(numerator), m_denominator(denominator)
{
}
//转换函数
operator double() const
{
return (double)m_numerator / m_denominator;
}
private:
int m_numerator; //分子
int m_denominator; //分母
};
int main()
{
Fraction f(3,5);
double sum = 4 + f;
std::cout << "sum = " << sum << std::endl; //sum = 4.6
return 0;
}
2.转换函数
转换函数的定义如下:
//转换函数
operator double() const
{
return (double)m_numerator / m_denominator;
}
operator是关键字,它通常和运算符一起使用,表示运算符函数。
double是函数名,表明了将Fraction转换为double类型。(正是由于此处指明了输出类型,所以该函数省略了返回值。编译器会根据函数名决定返回值类型)。
const是限制符,表示在该函数内,不会改变数据成员的值。(明确指定const,避免后期维护或重构时,出现错误)。
函数体内分子除以分母。由于两者均为整数,要使用doule强制转换,否则会返回整数。(为了简化,该代码省略了一些逻辑,如:分母不为0的判断)。
3.测试代码
调用转换函数的代码如下:
double sum = 4 + f;
当编译器看到这行代码时会做什么呢?
首先,它会寻找+操作符是否有重载了参数为double和Fraction的函数。很明显,我们并没有为刚刚新建的类Fraction重载+操作符,编译器的该次尝试以失败告终。
接着,不轻言放弃的编译器发现操作符+可以接收两个double类型的参数,那么是否可以将Fraction转换为double,从而调用该版本的函数呢?这次编译器成功了,因为我们刚刚实现了转换为double的函数。
最终,编译器成功实现了求和操作,输出了结果4.6。
4.扩展
我们来扩展一下转换功能。考虑能否将Fraction转化为其它类型,如string。
转换函数的定义如下:
//转换函数
operator string() const
{
return to_string(m_numerator) + "/" + to_string(m_denominator);
}
// 调用转换函数的代码如下:
string str = f ;
std::cout << "Result is: " << str << std::endl; //Result is: 3/5
代码的分析与转换为double类型相同,就不多叙述了。
2. non-explicit-one-argument ctor
另外一种直接讲某种类型的数据转换为类的方法,如果存在互相类型转换时,这个时候就需要explicit制定首先使用构造函数还是使用类型转换函数进行类型转换。
//分数类
class Fraction
{
public:
Fraction(int num,int den=1)
:m_numerator(num),m_denominator(den){}
//转换函数--> 把 Fraction 转换为double
operator double()const{
return (double)(m_numerator / m_denominator);
}
private:
m_numerator;
m_denominator;
};
//可能的用法 用法决定了类实现
Fraction f(3,5);
double d2=f+4;
class Fraction
{
public:
explicit Fraction(int num,int den=1)
:m_numerator(num),m_denominator(den){}
//转换函数--> 把 Fraction 转换为double
operator double()const{
return (double)(m_numerator / m_denominator);
}
Fraction operator+(const Fraction&f){
return Fraction(......);
}
private:
m_numerator;
m_denominator;
};
//可能的用法
Fraction f(3,5);
Fraction d2=f+4;//[Error] convirsion from 'double' to 'Fraction' requested
标准库中使用到的转换函数