C++ 类型转换函数 与 explicit

类型转换函数 与 explicit

1. 类型转换函数

C++中,可以使用构造函数将一个指定类型的数据转换为类的对象,也可以使用类型转换函数 (type conversion function)将一个类对象转换为其他类型的数据。

我们直接通过一个简单的代码介绍转换函数:

#include <iostream>
using namespace std;

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(void)
{
    Fraction f(3, 5); 
    double d = 3.2 + f;
    cout << d << endl;

    return 0;
}

我们设计了一个分数(Fraction)类,该类有两个私有变量,分别表示分子和分母。构造函数有两个参数,第二个参数默认为1。

在主函数中,定义了一个分数类的对象f,分子为3,分母为5。然后将3.2 + f的值设置为变量d

3.2是一个浮点类型的变量,而f是一个Fraction类型的变量,因此不能直接相加,于是编译器会到分数类中寻找,找到了operator double函数,该函数可以将类型对象转换为一个浮点类型的变量。因此该函数就是我们所说的类型转换函数(type conversion function)。通过分数类中的转换函数,我们给出转换函数的一般形式:

operator 类型名称() const {
        // 实现转换      
}
  1. 转换函数必须是类的成员函数
  2. 转换函数不能声明返回类型
  3. 形参列表必须为空
  4. 类型转换函数通常应该是const

类型转换运算符是隐式执行的,因此对象f就拥有了双重类型,既可以是类类型,也可以是double类型。

2. explicit

既然可以调用类型转换函数将该Fraction类型转换为double类型,那么也可以通过重载+double类型转换为类类型。因此该类的代码如下:

class Fraction
{
    public:
    Fraction(int num, int den = 1) : m_numerator(num), m_denominator(den) {}
    Fraction operator + (const Fraction& f) {
        return Fraction(f.m_numerator + this->m_numerator , f.m_denominator + this->m_denominator); 
    }

    private:
    int m_numerator;
    int m_denominator;
};
int main(void)
{
    Fraction f(3, 5); 
    Fraction d = f + 3.2 ;

    return 0;
}

当我们将f + 3.2的值赋值给d实例时,3.2就通过构造函数转换为Fraction类型,然后调用operator +函数将两个实例加起来赋值给d对象。

但是如果同时存在类型转换函数,会发生什么情况呢?我们加入类型转换函数:

#include <iostream>
using namespace std;

class Fraction
{
    public:
    /* explicit */ Fraction(int num, int den = 1) : m_numerator(num), m_denominator(den) {}
    /* explicit */operator double() const {
        return (double) m_numerator/m_denominator; 
    }
    Fraction operator + (const Fraction& f) {
        return Fraction(f.m_numerator+this->m_numerator, f.m_denominator+this->m_denominator); 
    }

    private:
    int m_numerator;
    int m_denominator;
};
int main(void)
{
    Fraction f(3, 5); 
    double d = f + 3.2;

    return 0;

} 

如果编译上述的代码就会立刻报错:ambiguous overload for ‘operator+’ (operand types are ‘Fraction’ and ‘double’)

因为存在了二义性。我们来具体分析以下:

  • f对象可以通过类型转换函数将类类型转换为double类型,然后加上3.2赋值给d
  • 3.2也可以通过构造函数将double类型转换为Fraction类型,然后两个类类型相加,又可以通过类型转换函数转换为double类型,所以这种方法也可以。

因此,以上两种方式都可以,所以编译器会报二义性的错误

解决的办法就是,使用explicit关键字的限制,使用方法也有两种:

  • 直接在构造函数前加上explicit关键字,防止double类型的变量隐式的转换为类类型。
  • 在转换构造函数operator double前加上explicit关键字,表示只有显示的将类类型转换为double类型时才调用该函数。
    • 例如:double d = static_cast<double>(f) + 3.2

后面的方法是C++11新引入的显示的类型转换运算符。

我们要尽量避免有二义性的类型转换,如果类中包含一个或多个类型转换,则必须确保在类类型和目标类型之间只存在唯一一种转换方式,否则将出现二义性。

  • 8
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: 我可以解释explicit构造函数的用法。 explicit构造函数C++中的一种特殊构造函数,它可以防止类的实例被隐式转换,从而避免出现错误。它通常被定义为一个具有const限定符的成员函数,这意味着它不能修改类的成员变量。 ### 回答2: 在C++中,构造函数被用于创建对象并初始化其成员变量。然而,有时我们可能希望禁止隐式地将某个类的一个参数类型转换为该类的对象类型,这时就可以使用explicit关键字来声明构造函数。 当我们使用explicit关键字声明一个构造函数时,它将不再能够进行隐式类型转换。只有在显式地调用构造函数时,才能使用该构造函数创建一个对象。 explicit关键字的使用可以避免一些潜在的问题和错误的发生。它可以防止编译器隐式地将一个参数类型转换为对象类型,以免引发意想不到的结果。 举个例子,假设有一个类A,其中有一个带有一个int参数的构造函数。如果我们不使用explicit关键字来声明该构造函数,那么在编译器看来,一个int类型的变量可以被自动转换为一个A类型的对象。这种隐式转换可能会导致意想不到的结果和错误的发生。 但是如果我们使用explicit关键字来声明该构造函数,那么编译器将不再允许隐式转换,只能通过显式调用构造函数来创建对象。 总之,explicit关键字用于禁止隐式地将一个参数类型转换为对象类型。它可以避免潜在的问题和错误,使代码更加可靠和健壮。 ### 回答3: 在C++中,构造函数(Constructor)是一种特殊类型的成员函数,用于在创建对象时初始化对象的数据成员。当我们定义一个类时,可以定义一个或多个构造函数来满足不同的对象创建需求。其中,explicit是一个关键字,用于修饰构造函数explicit关键字的作用是防止隐式转换,它只能用于单参构造函数(即只有一个参数的构造函数)。当构造函数声明为explicit时,禁止编译器通过隐式转换将该参数类型转换成对应的类类型。可以通过显式方式调用构造函数进行对象的创建和转换。 举个例子,假设有一个类A,其中定义了一个单参构造函数A(int n),同时使用了explicit关键字进行修饰。如果没有explicit关键字修饰,则可以进行隐式转换,比如可以使用A对象去初始化一个int类型的变量。但是,如果使用explicit修饰之后,编译器将不再允许隐式转换,只能通过显式方式进行构造和转换。 这种explicit用法可以避免一些不必要的类型转换带来的错误和混淆,能够增加程序的类型安全性。此外,explicit关键字还可以用于拒绝编译器进行多次隐式转换,因为如果构造函数没有使用explicit关键字修饰,编译器可能会进行多次自动类型转换,导致程序的行为变得复杂和难以理解。 总结来说,当我们在定义构造函数时,如果希望限制使用隐式转换创建对象的情况,就可以使用explicit关键字对构造函数进行修饰,以保证程序的可读性和类型安全性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值