【发布时间】:2013-12-12 02:00:24
【问题描述】:
这似乎是应该经常问和回答的问题,但我的搜索失败了。
我正在编写一个函数,我想接受某种通用的可调用对象(包括裸函数、手动仿函数对象、bind 或 std::function),然后在算法(即 lambda)。
函数当前声明如下:
template<typename T, typename F>
size_t do_something(const T& a, const F& f)
{
T internal_value(a);
// do some complicated things
// loop {
// loop {
f(static_cast<const T&>(internal_value), other_stuff);
// do some more things
// }
// }
return 42;
}
我通过引用接受函子,因为我想保证它不会在进入函数时被复制,因此实际上调用了对象的相同实例。它是一个 const 引用,因为这是接受临时对象的唯一方式(这在使用手动仿函数或 bind 时很常见)。
但这需要仿函数将operator() 实现为常量。我不想这样要求;我希望它能够接受两者。
我知道我可以声明此方法的两个副本,一个接受它作为 const,另一个接受它作为非常量,以涵盖这两种情况。但我不想这样做,因为 cmets 隐藏了很多我不想复制的代码(包括一些循环结构,所以我不能在不移动问题的情况下将它们提取到辅助方法) .
我也知道我可能会在调用它之前将仿函数 const_cast 欺骗为非常量,但这感觉有潜在的危险(特别是如果仿函数有意实现 const 和非常量,则会调用错误的方法呼叫操作员)。
我考虑过将函子作为std::function/boost::function 接受,但这对于本应是一个简单问题的问题来说似乎是一个沉重的解决方案。 (特别是在函子应该什么都不做的情况下。)
除了重复算法之外,是否有一种“正确”的方式来满足这些要求?
[注意:我更喜欢不需要 C++11 的解决方案,尽管我也对 C++11 的答案感兴趣,因为我在两种语言的项目中都使用了类似的结构。]
【问题讨论】:
-
嗯,按值获取可调用对象是很常见的。无论如何,为什么不将主代码放在另一个函数中,让 const/non-const 版本调用它并传递
f(...)的结果? -
在
C++11中,您将使用“通用参考”(F&&),请参阅 Scott Meyers 关于该主题的重要信息。 -
我有一条经验法则:当隐式转换有效时,切勿使用强制转换。 (关于你的
internal_value) -
@chris:如果我按价值接收它,它将被复制。这是不可接受的。
-
@BenVoigt:强制转换旨在“鼓励”仿函数以正确的方式接受参数,或者生成编译错误。特别是我想强制我想要的参数类型,即使仿函数也是模板化的。
标签: c++ constants functor function-object