我有一个非模板类NC和一个派生模板类TC。我想转换一个指向NC的指针,它可能是指向TC实例的指针,指向TC指针。模板的实际类型仅限于bool、int和string。
class NC {
...
}
template <typename T>
class TC: public NC {
private:
T value;
public:
...
void setValue(T value) {
this->value = value;
}
}
class UserValueProvider {
public:
int getValue() const { return 5; }
bool getValue() const { return true; }
string getValue() const { return "foobar"; }
}
void setUserValue(UserValueProvider *uvp, NC *obj) {
auto tobj = dynamic_cast< ? >(obj); // what goes here?
if(tobj)
tobj->setValue(uvp->getValue());
}显而易见的解决方案是执行3次动态强制转换(针对int、bool和string),并调用专用实例的setValue。然而,我想知道是否有另一种解决方案,因为可能的专业化越多,就越需要更多的动态强制转换,并且更有可能忘记一个专业化。
发布于 2020-02-28 18:42:14
这里的一种解决方案是翻转调用,并使用虚拟函数。但首先,让我们重写UserValueProvider::getValue,因为只对返回类型进行重载是被禁止的。
class UserValueProvider {
template <class T> struct tag { };
int getValue(tag<int>) const { return 5; }
bool getValue(tag<bool>) const { return true; }
std::string getValue(tag<std::string>) const { return "foobar"; }
public:
template <class T>
T getValue() const {
return getValue(tag<T>{});
}
};现在我们可以调用uvp.getValue<T>()来获取相应的值。接下来,向NC及其派生类添加一个虚函数:
class NC {
public:
virtual void setValueFrom(UserValueProvider &uvp) = 0;
};
template <typename T>
class TC: public NC {
private:
T value;
public:
void setValue(T value) {
this->value = value;
}
void setValueFrom(UserValueProvider &uvp) override {
setValue(uvp.getValue<T>());
}
};瞧,你只需将你的UserValueProvider传递给你的类型擦除的NC,它就会正确地分派。
void setUserValue(UserValueProvider *uvp, NC *obj) {
obj->setValueFrom(*uvp);
}https://stackoverflow.com/questions/60449853
复制相似问题