让函数返回C++类中的任何类型

sig*_*lor 2 c++

我有这门课:

#define TYPE_INVALID 0x00
#define TYPE_BYTE    0x01
#define TYPE_SHORT   0x02
#define TYPE_INT     0x03
#define TYPE_LONG    0x04
#define TYPE_FLOAT   0x05
#define TYPE_DOUBLE  0x06

class BASIC_TYPE
{
    private:
        int8_t  type;
        int8_t  byteValue;
        int16_t shortValue;
        int32_t intValue;
        int64_t longValue;
        float   floatValue;
        double  doubleValue;

    public:
        BASIC_TYPE();
        template<typename T> BASIC_TYPE(int8_t, T);

        template<typename T> void set(T);
        template<typename T> T    get();
};

BASIC_TYPE::BASIC_TYPE()
{
    type = TYPE_INVALID;
}

template<typename T> BASIC_TYPE::BASIC_TYPE(int8_t newType, T value)
{
    type = newType;
    set(value);
}

template<typename T> void BASIC_TYPE::set(T value)
{
    switch(type)
    {
        case TYPE_BYTE   : byteValue   = value; break;
        case TYPE_SHORT  : shortValue  = value; break;
        case TYPE_INT    : intValue    = value; break;
        case TYPE_LONG   : longValue   = value; break;
        case TYPE_FLOAT  : floatValue  = value; break;
        case TYPE_DOUBLE : doubleValue = value; break;
    }
}

template<typename T> T BASIC_TYPE::get()
{
    switch(type)
    {
        case TYPE_BYTE   : return byteValue;
        case TYPE_SHORT  : return shortValue;
        case TYPE_INT    : return intValue;
        case TYPE_LONG   : return longValue;
        case TYPE_FLOAT  : return floatValue;
        case TYPE_DOUBLE : return doubleValue;
    }
}
Run Code Online (Sandbox Code Playgroud)

现在我想使用get() - 函数输出存储的数字,如下所示:

BASIC_TYPE val1(TYPE_INT, 1234);
BASIC_TYPE val2(TYPE_DOUBLE, 3.1415926535);

val1.set(5678);
val2.set(2.7182818284);
printf("%d\n%f\n", val1.get(), val2.get());
Run Code Online (Sandbox Code Playgroud)

但G ++说有no matching function call to 'BASIC_TYPE::get()printf该-function和template argument deduction/substitution failed(couldn't deduce template parameter 'T').

需要更改什么才能让代码正确编译?

πάν*_*ῥεῖ 6

"需要更改什么才能让代码正确编译?"

在这种情况下,您需要明确.写吧

printf("%d\n%f\n", val1.get<int>(), val2.get<double>());
                        // ^^^^^            ^^^^^^^^
Run Code Online (Sandbox Code Playgroud)

C++不能get()仅仅通过返回类型来区分各种函数实例化(其他可以与set()where T出现的参数类型一样).
在这种情况下,您需要指定类型以明确地实例化模板化函数(如上所示).


另请注意:我更愿意使用enum而不是那些#define TYPE_xxx语句:

enum MySupportedTypes {
    TYPE_INVALID ,
    TYPE_BYTE    ,
    TYPE_SHORT   ,
    TYPE_INT     ,
    TYPE_LONG    ,
    TYPE_FLOAT   ,
    TYPE_DOUBLE  ,
};
Run Code Online (Sandbox Code Playgroud)

对于get()模板函数的实现,您应该考虑类似这样的1,以避免get()为不适当的请求类型调用函数.

template<typename T> T BASIC_TYPE::get() {
    switch(type) {
        case TYPE_BYTE: 
            std::is_same<T,int8_t> ? return byteValue : throw std::bad_typeid;
        case TYPE_SHORT: 
            std::is_same<T,int16_t> ? return shortValue : throw std::bad_typeid;
        //  analogous ...
    }
}
Run Code Online (Sandbox Code Playgroud)

或者甚至更好地提供一种在编译时捕获类型不匹配的机制.


1)参见文档参考资料 std::is_same