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


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

    如果是成员函数:

    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

  • 相关阅读:
    003.同时Ping多个IP(select实现IO复用,信号计时),ping程序升级版
    002.ICMP--拼接ICMP包,实现简单Ping程序(原始套接字)
    001.linux下clock()检测程序运行时间
    django form的函数用法
    命令注入利用语句
    小白审计JACKSON反序列化漏洞
    代码审计小工具
    Burp插件开发--应用篇
    burp插件开发--基础篇
    JAVA web网站代码审计--入门
  • 原文地址:https://www.cnblogs.com/Przz/p/6659236.html
Copyright © 2020-2023  润新知