0

我有一个用于创建类的工厂方法模式,我想在进行实际创建之前验证输入参数。我在这里处理两个类,A 和 B,它们都有自己的验证。鉴于我已经在创建模板函数中传递了类类型(A 或 B),我想直接使用该类型来告诉我需要使用哪个验证。我正在考虑使用模板函数专业化或虚拟输入参数来做到这一点,每个都有自己的优点/缺点。如果其中任何一个看起来更合理,或者是否还有其他选择,请告诉我。谢谢!

哦,我确实考虑过将验证作为 A 或 B 的静态方法,但这看起来不太好,因为我正在处理遗留代码并且移动这些东西并不是太简单。为了完整起见,我还在下面将其显示为一个选项。

模板功能特化:

优点:可以直接使用传入 FactoryCreateClass 方法的类型来选择验证

缺点:没有利用模板方法,因为只有专业化

template <typename T>
void FactoryCreateClass (int, double)
(
    bool bSuccess = ValidateBlock<T>(int, double);

    if (bSuccess)
        T* = T::CreateInstance();           
)

template <typename T>
bool ValidateBlock (int double);    // don't need an implementation here since 
                                    // all specialization require different validation

template <>
bool ValidateBlock<A> (int, double)
{
    //do validation before creating A
}

template <>
bool ValidateBlock<B> (int, double)
{
    //do validation before creating B
}

虚拟变量:

优点:可以直接使用传入 FactoryCreateClass 方法的类型来选择验证

缺点:未使用的虚拟变量

template <typename T>
void FactoryCreateClass (int, double)
(
    bool bSuccess = ValidateBlock(T /*dummy*/, int, double);

    if (bSuccess)
        T* = T::CreateInstance();           
)

bool ValidateBlock (A/*dummy*/, int, double)
{
    //do validation before creating A
}

bool ValidateBlock (B/*dummy*/, int, double)
{
    //do validation before creating B
}   

静态类方法:

优点:可以直接使用传入 FactoryCreateClass 方法的类型来选择验证

缺点:更难在遗留代码中进行这种类型的更改

template <typename T>
void FactoryCreateClass (int, double)
(
    bool bSuccess = T::ValidateBlock(int, double);

    if (bSuccess)
        T* = T::CreateInstance();           
)

static bool A::ValidateBlock (int, double)
{
    //do validation before creating A
}

static bool B::ValidateBlock (int, double)
{
    //do validation before creating B
}
4

2 回答 2

1

您忘记了第三种选择:使用特征。

template <typename> struct Trait;

template <>
struct Trait<A> {
    static bool Validate(int, double);
};

template <>
struct Trait<B> {
    static bool Validate(int, double);
};

接着:

template <typename T>
void FactoryCreateClass(int i, double d) {
    bool const success = Trait<T>::Validate(i, d);
    // ...
}

但当然,真正的问题是为什么不CreateInstance直接进行验证?

于 2013-02-24T13:58:12.503 回答
1

这里最合适的设计是在CreateInstance方法内部进行验证,因为任务(args 验证和实例创建)是强耦合的,将这些东西放在一个地方更具可读性。

但是,可能是这种情况,您不能更改CreateInstance方法的代码。那么,我发现使用函数模板或虚拟变量之间没有显着差异。然而,后者更加明确,因此更具可读性。

顺便说一句,一个未引用的变量真的不是什么大问题,你可以抑制它,见 Disable single warning errorUNREFERENCED_PARAMETER ,在页面上搜索

于 2013-02-24T16:24:40.180 回答