在C ++中处理可选参数/默认值的最佳方法

时间:2015-04-01 03:14:46

标签: c++ default-value optional-parameters

我想知道是否可以通过以下方式执行以下操作:

    template <typename T>
void _class_<T>::func (T chgx = x, T chg y = y, T chgz = z)
{
     x = chgx;
     y = chgy;
     z = chgz;
}

我想要做的是将默认值设置为我想要更改的值,这样如果我省略一些参数,那些值不会被更改。上面的代码不起作用,我想我理解为什么(函数实际上没有访问这些值,因此它会抛出错误)。

我想知道的是,是否有办法像我上面描述的那样,所有这些都有一个功能。我唯一的选择是使用多个定义重载函数来处理不同数量的参数?

3 个答案:

答案 0 :(得分:1)

您可以通过拥有一组重载函数来完成您想要的任务。

// First overload. Needs all arguments.
template <typename T>
void _class_<T>::func (T chgx, T chgy, T chgz)
{
     x = chgx;
     y = chgy;
     z = chgz;
}

// Second overload. Needs two arguments.
template <typename T>
void _class_<T>::func (T chgx, T chgy)
{
     x = chgx;
     y = chgy;
}

// Third overload. Needs one argument.
template <typename T>
void _class_<T>::func (T chgx)
{
     x = chgx;
}

// You can have a fourth overload that takes no arguments.
// However, it's totally useless.
template <typename T>
void _class_<T>::func ()
{
   // Nothing needs to be changed.
}

代码重复较少的代码(感谢@BenVoigt提供的建议):

// Second overload. Needs two arguments.
template <typename T>
void _class_<T>::func (T chgx, T chgy)
{
   func(chgx, chgy, z);
}

// Third overload. Needs one argument.
template <typename T>
void _class_<T>::func (T chgx)
{
   func(chgx, y);
}

// You can have a fourth overload that takes no arguments.
// However, it's totally useless.
template <typename T>
void _class_<T>::func ()
{
   func(x);
}

答案 1 :(得分:1)

一种方法是使用boost::optional这是一个小包装器,如果没有给出参数,则默认为空:

using boost::optional;

template<typename T>
void _class_<T>::func( optional<T> cx = {}, 
     optional<T> cy = {}, optional<T> cz = {} )
{
    if ( cx ) x = *cx;
    if ( cy ) y = *cy;
    if ( cz ) z = *cz;
}

虽然你没有得到移动或引用语义(所以我仍然更喜欢R Sahu的解决方案)

答案 2 :(得分:1)

一种方式(同样,不支持移动或引用语义)是使函数通过初始化列表获取变量参数:

template<typename T>
void _class_<T>::func( std::initializer_list<T> items )
{
    T const *ptr = items.begin();

    if ( ptr != items.end() ) x = *ptr++;
    if ( ptr != items.end() ) y = *ptr++;
    if ( ptr != items.end() ) z = *ptr++;
}

用法:

obj.func({ 1, 2, 3 });
obj.func({ 1 });
obj.func({});