【发布时间】:2009-11-18 17:43:13
【问题描述】:
碰到另一个模板问题:
问题:对于对象是指针的情况,我想部分专门化一个容器类 (foo),并且我只想专门化删除方法。应该是这样的:
库代码
template <typename T>
class foo
{
public:
void addSome (T o) { printf ("adding that object..."); }
void deleteSome (T o) { printf ("deleting that object..."); }
};
template <typename T>
class foo <T *>
{
public:
void deleteSome (T* o) { printf ("deleting that PTR to an object..."); }
};
用户代码
foo<myclass> myclasses;
foo<myclass*> myptrs;
myptrs.addSome (new myclass());
这导致编译器告诉我 myptrs 没有名为 addSome 的方法。 为什么?
谢谢。
Solution
基于 tony 的回答,这里是完全可编译的东西库
template <typename T>
class foobase
{
public:
void addSome (T o) { printf ("adding that object..."); }
void deleteSome (T o) { printf ("deleting that object..."); }
};
template <typename T>
class foo : public foobase<T>
{ };
template <typename T>
class foo<T *> : public foobase<T *>
{
public:
void deleteSome (T* o) { printf ("deleting that ptr to an object..."); }
};
用户
foo<int> fi;
foo<int*> fpi;
int i = 13;
fi.addSome (12);
fpi.addSome (&i);
fpi.deleteSome (12); // compiler-error: doesnt work
fi.deleteSome (&i); // compiler-error: doesnt work
fi.deleteSome (12); // foobase::deleteSome called
fpi.deleteSome (&i); // foo<T*>::deleteSome called
【问题讨论】:
-
错误的解决方案。假设我们使用 foo
。假设 class foobase的某个函数调用了deleteSome。会调用什么函数?正确的!class foobase的 deleteSome(T) 而不是 deleteSome(T *)。 -
@Malistov:我认为关键是不应该使用 foobase。它只是为了使黑客成为可能。
-
@splicer,但如果您调用
deleteSome以外的函数,它将隐式使用。到那时,如果被调用函数调用deleteSome,它将永远无法到达派生类中的函数。 -
@JohannesSchaub-litb 如果基类指定函数是虚拟的(并且没有实例化)怎么办?我以前见过这个问题/设计模式:它有名字吗?
标签: c++ templates partial-specialization