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

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

如果是成员函数:

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

转载于:https://www.cnblogs.com/Przz/p/6659236.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值