模板的非成员函数 和 类型转换

若所有参数都需要类型转换:

如果是成员函数:

Retional a(1,4);
Retional result;

result = a*2;  //通过隐式转换
retult = 2*a;  //error

因为上面的两局可以转换成:

result = a.operator*(2);
result = 2.operator*(a);

如果构造函数是explicit的,那么两种都会失败

所以通常会被弄成一个非成员函数:

const Rational operator*(const Rational& a,const Rational& b) {}

至于是否应该是friend则需要看情况。


模板template:

但是上面的方法在涉及到模板的时候会出现问题。

template<typename T>
class Rational{
public:
    Rational(const T& numerator=0,denominator=1);
    const T numerator() const;
    const T denominator()const;
    ...
};
template<typename T>
const Rational<T> operator*(const Rational<T>& lhs,
                                             const Rational<T>& rhs){
    ...
}

Rational<int>results = oneHalf * 2 ; //error

我们会希望编译器能够通过隐式转换将 int 转换成 Rational< int >,进而得到T = int。
实际上模板的实参推导从来不将隐式转换函数纳入考虑范围之内(尽管隐式转换在函数调用的过程中会被使用)。
所以无法找到对应的模板进行是具现化,而在具现化之前不考虑隐式转换。

当编译器遇到一个模板定义时,它并不生成代码。只有我们实例化出模板的一个特定版本时,编译器才会生成代码。
C++函数匹配时会在候选函数中找到参数相同 或者 能进行隐式转换的。但是模板没有实例化,也就找不到那函数了。


解决方案:

可以将operator* 模板函数声明成Rational< T >的友元。
那么在Rational< int >具现化的时候,作为其一部分的operator*也就被自动声明出来的。因此在后面能够 通过隐式转换匹配到它。

template<typename T>
class Rational{
public:
    friend Rational operator*(const Rational& lhs,
                                           const Rational& rhs);
    ...
};
template<typename T>
const Rational<T> operator*(const Rational<T>& lhs,
                                             const Rational<T>& rhs){
    ...
}

通常在模板类里面外面可以省去< T >,但是上面类内部的operator*并不是一个函数模板,只能说是一个依赖于模板参数的普通函数,所以外面的定义式和它本质上并不是同一类东西,也就链接不到它。

  • 可以考虑在前面先将相应的operator定义成模板函数再处理
  • 在类内部实现定义(因为依赖于参数T,所以类模板的每个实例化都要单独定义这个函数)

参考:

《Effective C++》
《C++primer 5》
关于c++中模板类中友元函数的定义和使用:http://bbs.csdn.net/topics/391867303

posted @ 2017-04-02 16:18  Przz  阅读(423)  评论(0编辑  收藏  举报