这是我的代码:
struct S
{
int f() { return 1; }
int g(int arg = f()) { return arg; }
};
int main()
{
S s;
return s.g();
}
无法使用错误进行编译:
error: cannot call member function 'int S::f()' without object
尝试this->f()
也不起作用,因为this
可能不会在该上下文中使用。
有没有办法让这项工作仍然使用默认参数?
当然可以通过不使用默认参数来解决它:
int g(int arg) { return arg; }
int g() { return g(f()); }
然而,考虑到在“真实代码”中arg
之前有更多参数,以及遵循此模式的几个函数,这会变得冗长。 (如果在一个函数中有多个默认参数,那就更难看了。)
NB。 This question一开始看起来很相似,但事实上他正在询问如何形成一个闭包,这是一个不同的问题(并且链接的解决方案不适用于我的情况)。
答案 0 :(得分:15)
如果他们是static
,您只能使用其中的成员。从C ++ 11草案(n3299),§8.3.6/ 9:
类似地,非静态成员不应在默认参数中使用,即使它未被计算,除非它显示为类成员访问表达式的 id-expression (5.2。 5)或除非是 用于形成指向成员的指针(5.3.1)。
e.g。这有效:
struct S {
static int f() { return 1; }
int g(int arg = f()) { return arg; }
};
int main()
{
S s;
return s.g();
}
这也有效(我认为这是第一个表达的含义):
struct S {
int f() { return 42; }
int g(int arg);
};
static S global;
int S::g(int arg = global.f()) { return arg; }
int main()
{
S s;
return s.g();
}
至于this
,确实不允许(§8.3.6/ 8):
关键字
this
不得用于成员函数的默认参数。
cppreference.com上的default arguments页面有很多关于子喷射的详细信息 - 它可能会非常复杂。
答案 1 :(得分:5)
如果允许您使用C ++ 17中的实验性功能,则可以使用STL中的std::optional
(有关详细信息,请参阅here)。
在其他方面类似于:
int g(std::optional<int> oarg = std::optional<int>{}) {
int arg = oarg ? *oarg : f();
// go further
}
修改强>
正如评论中所建议的那样,上面的代码在逻辑上应该等同于下面的代码:
int g(std::optional<int> oarg = std::optional<int>{}) {
int arg = oarg.value_or(f());
// go further
}
这个更具可读性(不是吗?),但请注意它在任何情况下都会执行f
。
如果这个功能很昂贵,也许它不值得。
答案 2 :(得分:3)
我添加了另一个答案,与前一个完全不同,可以解决您的问题 我们的想法是使用另一个类以及显式和非显式构造函数的正确组合 它遵循一个最小的工作示例:
#include <functional>
#include <iostream>
template<class C, int(C::*M)()>
struct Arg {
std::function<int(C*)> fn;
Arg(int i): fn{[i](C*){ return i; }} { }
explicit Arg(): fn{[](C* c){ return (c->*M)(); }} { }
};
struct S {
int f() { return 1; }
int h() { return 2; }
void g(int arg0,
Arg<S, &S::f> arg1 = Arg<S, &S::f>{},
Arg<S, &S::h> arg2 = Arg<S, &S::h>{})
{
std::cout << "arguments" << std::endl;
std::cout << "arg0: " << arg0 << std::endl;
std::cout << "arg1: " << arg1.fn(this) << std::endl;
std::cout << "arg2: " << arg2.fn(this) << std::endl;
}
};
int main() {
S s{};
s.g(42, 41, 40);
s.g(0);
}
该示例显示了如何混合默认参数和非默认参数
修改它非常简单,让g
成为具有空参数列表的函数,如原始问题所示。
我也很确定一个人可以改进这个例子并以更好的方式结束,无论如何它应该是一个很好的起点。
它遵循从问题中应用于原始示例的解决方案:
#include <functional>
template<class C, int(C::*M)()>
struct Arg {
std::function<int(C*)> fn;
Arg(int i): fn{[i](C*){ return i; }} { }
explicit Arg(): fn{[](C* c){ return (c->*M)(); }} { }
};
struct S {
int f() { return 1; }
int g(Arg<S, &S::f> arg = Arg<S, &S::f>{}) {
return arg.fn(this);
}
};
int main() {
S s{};
return s.g();
}
就是这样,即使没有static
方法或全局变量,也可以这样做。
当然,我们可以以某种方式使用我们的 this 。这是一个弯曲语言的问题......