转换构造函数和类型转换运算符共同定义了类类型转换,也被称为用户定义的转换

类型转换运算符
是类的一种特殊成员函数,一般形式为operator type() const,类型转换运算符可以面向任意类型(除了void)进行定义,只要该类型能作为函数的返回类型。因此,不允许转换成函数和数组类型,但是可以转换成指针或者引用
类型转换运算符没有显式的返回类型,也没有形参,必须定义成类的成员函数,而且通常不改变内容,定义成const

class SmallInt
{
public:
    SmallInt(int i = 0):val(i)
    {
        if(i < 0 || i > 255)
            throw std::out_of_range("Bad SmallInt value");
    }
    operator int() const
    {return val;}
private:
    std::size_t val;
};

显式的类型转换运算符
通过在前面使用explicit,可以定义显式的类型转换运算符,编译器不会将显式的类型转换运算符用于隐式转换,除非显式的使用强制类型转换来请求
另外,当表达式被用于条件时,显式的可能会被隐式的调用

流对象转换为bool
之所以能使用while(cin>>val),因为cin被istream类的operator bool类型转换运算符进行了转换,如果cin的状态为真,就会返回bool值真,从而用于判断

实参匹配和相同的类型转换
当存在多种类型转换时,尽量使用显式的调用方法,否则编译器难以区分要使用哪种转换
尽量不要定义多种依赖内置算术类型的转换,编译器将根据标准类型转换的优劣进行匹配,如果一样优劣,就会产生二义性

函数匹配与重载运算符
表达式中运算符的候选函数集既应该包括成员函数,也应该包括非成员函数

class SmallInt
{    
    friend SmallInt operator+(const SmallInt&, const SmallInt&);
public:
    SmallInt(int = 0);
    operator int() const
    {return val;}
private:
    std::size_t val; 
};
//可以使用上面的类将两个SmallInt相加,但是如果使用混合运算会出现二义性
SmallInt s1, s2;
SmallInt s3 = s1+s2;
int i = s3 + 0;//错误,可以先把0转成SmallInt再计算,也可以把s3转换成Int再计算,产生了二义性