C ++铸造模板类

时间:2010-10-30 00:44:33

标签: c++ templates gcc casting

我对模板有疑问。我想有一个模板化的类,其中包含一个float或double的数组。

我可以编写一个复制它的clone()函数。没问题。但是,我想有另一个名为cast()的函数,它在double和float之间来回转换。之前已经对此进行了一些讨论,但我不认为问题是一样的:

stackoverflow.com/questions/714213/c-template-casting

我遇到的问题是编译器错误,而不是链接器错误。错误消息是:

main.cpp: In function `void caster()':
main.cpp:63: error: expected primary-expression before "double"
main.cpp:63: error: expected `;' before "double"
main.cpp:64: error: expected primary-expression before "float"
main.cpp:64: error: expected `;' before "float"
main.cpp:65: error: expected primary-expression before '>' token
main.cpp:65: error: expected primary-expression before ')' token

我转了下面的代码。第63,64和65行是我在这里评论“Error here”的地方。

顺便说一句,我的编译器是GNU C版本3.4.5 20051201编译的“GNU C ++版本3.4.5 20051201(Red Hat 3.4.5-2)(x86_64-redhat-linux)”(Red Hat 3.4.5- 2)”。

经过一些谷歌搜索,结果发现有人已经遇到过这个问题:

gcc.gnu.org/ml/gcc-help/2006-04/msg00022.html

这里有一个解决方案:

gcc.gnu.org/ml/gcc-help/2006-04/msg00023.html

但是当原始海报询问为什么时,答案不是很明确:

gcc.gnu.org/ml/gcc-help/2006-04/msg00025.html

不幸的是,链接已经死了,我没有第三版Stroustrup。现在,我有我的修复,我的代码工作。但是,Stackoverflow,为什么是否有效?


#include <stdio.h>

// =================== This would be the header ===================
template <class T>
class foo
{
public:
             foo(const T val) {d_data = new double; *d_data = val;}
    virtual ~foo() {delete d_data;};

    foo* clone() const;

    template<class U>
    foo<U>* cast() const;

private:
    double *d_data;
};

// =================== This would be the implementation of the class ===================
template<class T>
foo<T>* foo<T>::clone() const
{
    return new foo<T>(*d_data);
}

template<class T>
template<class U>
foo<U>* foo<T>::cast() const
{
    return new foo<U>(*d_data);
}

template class foo<float>;
template class foo<double>;

template foo<float>* foo<float>::cast() const;
template foo<float>* foo<double>::cast() const;
template foo<double>* foo<float>::cast() const;
template foo<double>* foo<double>::cast() const;

// =================== Using the class ===================
template <class T>
void caster()
{
    foo<double> *f1 = NULL;
    foo<float>  *f2 = NULL;
    foo<T>      *f3 = NULL;

    // I am looking at something that compiles
    // I don't care about linking for now
    // This will crash at runtime because of
    // NULL, but that's just an example

    f1->cast<double>(); // compiler OK
    f1->cast<float>();  // compiler OK
    f1->cast<T>();      // compiler OK

    f2->cast<double>(); // compiler OK
    f2->cast<float>();  // compiler OK
    f2->cast<T>();      // compiler OK

    f3->cast<double>(); // Error here
    f3->cast<float>();  // Error here
    f3->cast<T>();      // Error here

    f3->foo<T>::template cast<double>(); // It works!
    f3->foo<T>::template cast<float>();  // It works!
    f3->foo<T>::template cast<T>();      // It works!
}

int main(int argc, char **argv)
{
    return 0;
}

1 个答案:

答案 0 :(得分:4)

f3->cast<double>(); // Error here 

在这一行中,编译器不知道<之后的f3->cast是否应该表示模板参数的开头,或者它是否小于比较运算符。

明确指出它代表模板参数的开头。正确的方法是

f3->template cast<double>();

总之,.template表示法(以及类似表示法,例如->template)只应在模板内使用,并且只有在它们遵循依赖于模板参数的内容时才会使用(例如表达式{{ 1}}取决于模板参数f3