2

我正在尝试制作一种可以类型安全地封装任意类型的类型。我从这个答案中得到了这样的想法:5 年后,有没有比“最快的 C++ 代表”更好的东西?到目前为止,我只成功地解决了这个问题,但是我遇到了一个我找不到根源的错误。

编译器似乎告诉我它不能将值转换为值自己的类型,这让我觉得很奇怪。

我正在运行带有 llvm-gcc 4.2(gcc 4.2.1 前端)的 Mac OS X 10.6。

欢迎提出如何摆脱 void* 或将其移至不太重要的位置的建议,但这个问题并不是真正的问题。

错误:

$ g++ main.cpp
main.cpp: In static member function ‘static Stamp StampFactory<T>::make(T*) [with T = int]’:
main.cpp:33:   instantiated from ‘Stamp makeStamp(T*) [with T = int]’
main.cpp:39:   instantiated from here
main.cpp:26: error: could not convert template argument ‘t’ to ‘int*’

编码:

typedef void (*VoidFunc)(void*);

struct Stamp
{
    Stamp(VoidFunc p)
    {
        this->press = p;
    }
    VoidFunc press;
};

template<typename T>
struct StampFactory
{
    template<T* rvalue>
    struct Pattern
    {
        void operator()(void* lvalue)
        {
            *dynamic_cast<T*>(lvalue) = *rvalue;
        }
    };

    static Stamp make(T* t)
    {
        return Stamp(Pattern<t>()); // 28
    }
};

template<typename T>
Stamp makeStamp(T* t)
{
    return StampFactory<T>::make(t); // 33
}

int main(int argc, char** argv)
{
    int i = 0;
    Stamp s = makeStamp(&i); //39
}
4

1 回答 1

4

该错误是由于模板参数必须是编译时常量(或constexpr),因此不能是变量(或函数参数)。允许将指针作为模板参数,但您可以提供的内容不多,因为它需要是编译时常量指针值(我能想到的唯一符合条件的是 char -指向字符串文字的指针)。一般规则很简单:所有模板参数必须在编译时已知,无论是类型还是值。这不包括函数参数或其他类型的运行时变量。

我希望我能提出一个替代方案来实现你想要的,但我根本无法理解你实际上想要做什么。

于 2012-09-14T15:29:25.270 回答