3

我正在尝试附加要在销毁模板化类时调用的函数对象。但是,我似乎无法将函数对象作为临时对象传递。我得到的警告是(如果注释行xi.data = 5;):

    warning C4930: 'X<T> xi2(writer (__cdecl *)(void))': 
    prototyped function not called (was a variable definition intended?)
            with
            [
                T=int
            ]

如果我尝试使用构造的对象,则会收到编译错误消息:

error C2228: left of '.data' must have class/struct/union

对于这段冗长的代码,我深表歉意,但我认为所有组件都需要可见才能评估情况。

template<typename T>
struct Base
{
    virtual void run( T& ){}
    virtual ~Base(){}
};

template<typename T, typename D>
struct Derived : public Base<T>
{
    virtual void run( T& t )
    {
        D d;
        d(t);
    }
};

template<typename T>
struct X
{
    template<typename R>
    X(const R& r)
    {
       std::cout << "X(R)" << std::endl;
       ptr = new Derived<T,R>(); 
    }

    X():ptr(0)
    { 
        std::cout << "X()" << std::endl; 
    }

    ~X()
    {
        if(ptr) 
        {
            ptr->run(data);
            delete ptr;
        }
        else
        {
            std::cout << "no ptr" << std::endl;
        }
    }

    Base<T>* ptr; 
    T data;
};

struct writer
{
    template<typename T>
    void operator()( const T& i )
    { 
        std::cout << "T : " << i << std::endl;
    }
};

int main()
{
    {
        writer w;
        X<int> xi2(w);
        //X<int> xi2(writer()); //This does not work!
        xi2.data = 15;       
    }

    return 0;
};

我尝试这样做的原因是,我可以“以某种方式”将函数对象类型与对象附加在一起,而无需在类中保留函数对象本身的实例。因此,当我创建 的对象时class X,我不必在class writer其中保留一个对象,而只需一个指向Base<T>(我不确定我是否需要<T>这里,但现在它在那里)的指针。

问题是我似乎必须创建一个对象,writer然后将其传递给构造函数,X而不是像这样调用它X<int> xi(writer();

我可能在这里遗漏了一些完全愚蠢和明显的东西,有什么建议吗?

4

3 回答 3

7

看起来像一个“最令人烦恼的解析”问题。尝试

X<int> xi2 = X<int>(writer());

或者

X<int> xi2((writer()));
于 2010-04-06T13:21:00.803 回答
5

尝试在这周围添加一对额外的括号,X<int> xi2((writer()));这将阻止编译器认为您预先声明了一个函数。(Scott Meyers 有效的 STL 项目 6。)

于 2010-04-06T13:20:42.517 回答
5

X<int> xi2(writer());是一个名为 xi2 的函数的声明,它本身返回一个X<int>,并将一个不带参数并返回一个 writer 的函数作为参数。这是一个“最令人烦恼的解析”。

解决方案是要么做你所做的,避免临时的,要么添加更多的括号。

于 2010-04-06T13:21:31.677 回答