- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
假设我有一个使用 std::function
实现策略模式的计算器类对象如下(参见 Scott Meyers,Effective C++:改进程序和设计的 55 种具体方法):
class Calculator
{
public:
...
std::vector<double> Calculate(double x, double y)
{
std::vector<double> res;
for(const Function& f : functions)
res.push_back(f(x,y));
return res;
}
private:
std::vector<Function> functions;
};
在哪里
typedef std::function<double(double,double)> Function;
这是我面临的问题:假设函数 f
和 g
, 都是 Function
类型,在内部执行昂贵且相同的计算以获得最终结果。为了提高效率,可以将所有公共(public)数据包装在一个 struct
中。 ,计算一次并作为参数提供给他们。然而,这种设计有几个缺陷。例如,这会导致 Function
的签名发生变化。 ,这可能导致将不必要的参数传递给某些函数实现。此外,这些公共(public)和内部数据不再对代码中的其他组件隐藏,这可能会损害代码的简洁性。
我想讨论以下优化策略:实现一个类 CacheFG
那:
Update
用给定的 double 计算其内部数据的方法 x
和 y
;和Check
确定其当前内部数据是否使用给定 double 计算的方法 x
和 y
.然后我们可以做的就是制作f
和 g
共享类的公共(public)实例 CacheFG
,这可以使用 std::shared_ptr
来完成构造。因此,下面将创建 f
和 g
使用辅助函数的函数 f_aux
和 g_aux
.
double f_aux(double x, double y, const std::shared_ptr<CacheFG>& cache)
{
if(not cache->Check(x,y))
cache->Update(x,y);
...
}
std::shared_ptr<CacheFG> cache;
Function f = std::bind(f_aux, _1, _2, cache);
Function g = std::bind(g_aux, _1, _2, cache);
我的问题是:(1) 这是一种安全的优化方法吗? (2) 有没有更好的方法来解决这个问题?
编辑:经过几个回答后,我发现我的目的是在 C++ 中实现一个内存技术。我注意到只有最后计算的状态就足以满足我的目的。
感谢 DeadMG,我现在将在这里写下对他的方法的改进。他的想法包括使用内存技术 variadic templates .我只是提供了一个细微的修改,我在其中使用了结构 std::decay<Args>::type
确保定义一个tuple
仅适用于非引用类型。否则,带有 const-reference 参数的函数会导致编译错误。
template<typename Ret, typename... Args>
std::function<Ret(Args...)> MemoizeLast(std::function<Ret(Args...)> f)
{
std::tuple<typename std::decay<Args>::type...> cache;
Ret result = Ret();
return [=](Args... args) mutable -> Ret
{
if(std::tie(args...) == cache)
return Ret(result);
cache = std::make_tuple(args...);
return result = f(args...);
};
}
为了防止result
的移动, 当提供 return Ret(result)
时返回它的拷贝 ( args
)是缓存的。
最佳答案
为什么要创建自己的类?您无需重新创建unordered_map
的界面。此功能可以添加为基于 std::function
和 std::unordered_map
的可重用算法。我已经有一段时间没有使用可变参数模板了,但我希望你能理解。
template<typename Ret, typename... Args>
std::function<Ret(Args...)> memoize(std::function<Ret(Args...)> t) {
std::unordered_map<std::tuple<Args...>, Ret> cache;
return [=](Args... a) mutable -> Ret {
if (cache.find(std::make_tuple(a...)) != cache.end())
return cache[std::make_tuple(a...)];
else
return cache[std::make_tuple(a...)] = t(a...);
};
}
我不记得 std::hash
是否原生支持元组。如果没有,您可能需要添加它,或使用原生支持它们的 std::map
。
编辑:嗯,我没注意到你想要共享缓存。好吧,这应该不是一个太难的问题,只需在计算器中粘贴一个 unordered_map
成员并通过引用传递它,但这样做的语义似乎有点……奇怪。
再次编辑:只是最近的值?更简单。
template<typename Ret, typename... Args>
std::function<Ret(Args...)> memoize_last(std::function<Ret(Args...)> t) {
std::tuple<Args...> cache;
Ret result;
return [=](Args... a) mutable -> Ret {
if (std::tie(a...) == cache)
return result;
cache = std::make_tuple(a...);
return result = t(a...);
};
}
如果你想在几个函数之间共享,那么改变是一样的 - 只需在类中声明它并作为引用传递。
关于c++ - 缓存优化和多态性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13066756/
我在 Mac OsX 10.11 上使用 Xcode 7.0.1 (7A1001) 我使用 carthage 0.9.2 通过以下购物车文件下载reactivecocoa github“Reactiv
我正在将一个对象从属性“模型”(我从 Laravel 中的 Blade 属性模型中获得)分配给数据属性模型。后来数据属性模型发生变化,因为它绑定(bind)到表单输入字段。但 Prop “模型”也发生
当我更新数组内对象的属性然后作为组件的 Prop 传递时,在 svelte 中触发 react 性的正确方法是什么? let items = [{ id: 1, name: 'first'
我是 DRY principle 的坚定拥护者: Every piece of knowledge must have a single, unambiguous, authoritative rep
我正在实现一个需要以下功能的线程: 及时响应终止请求 推送消息 在等待消息时保持对 SendMessage 请求的响应 我对消息泵的初始实现使用了 GetMessage,如下所示: while not
在我的应用程序中,用户获得了一份已到达她的文档列表,并且可以对每个文档执行操作。 文件是分批提交的,当这种情况发生时,列表会增加。这一切都很好,这是预期的行为,但最好有一个按钮“暂停实时数据”,它会忽
我有一个属性为 的数据对象 displaySubtotal 我可以通过以下方式更新该属性的值: data.displaySubtotal = numPad.valueAsAString(); 我的方法
我需要一个垂直 slider 输入。由于内置的 sliderInput 函数无法做到这一点,因此我选择自己实现。根据this thread可以 (I) 使用 CSS 旋转 sliderInput
我正在从自定义用户权限管理系统迁移到 Alanning:roles v2.0 .我有一个非常基本的结构: 基本用户 用户组,每个用户组都有特定的设置。我将它们存储在一个“组”集合中。 管理群组的用户的
Shiny 中的响应式(Reactive)表达式将更改传播到需要去的地方。我们可以使用 isolate 来抑制一些这种行为。 ,但是我们可以抑制基于我们自己的逻辑表达式传播的更改吗? 我给出的例子是一
是否有(或可能有) react 性 Parsec (或任何其他纯函数式解析器)在 Haskell 中? 简而言之,我想逐个字符地为解析器提供数据,并获得与我提供的足够多的结果一样多的结果。 或者更简单
HTML(JADE) p#result Lorem ipsum is javascript j s lo 1 2 4 this meteor thismeteor. meteor input.sear
我有一个被导入函数更改的对象。 https://svelte.dev/repl/e934087af1dc4a25a1ee52cf3fd3bbea?version=3.12.1 我想知道如何使我的更改反
我有一个YUV 420半平面格式的图像,其中字节以这种方式存储: [Y1 Y2 ... [U1 V1.... Yk Yk+1...] Uk' Uk'+1] 其中Y平面的大小是UV平面的两倍,并
如何使用 ReactiveCocoa 订阅从 NSMutableDictionary 添加和删除的对象?另外,我想在它发生变化时广播通知。我的猜测是可以使用 RACMulticastConnectio
我正在构建一个带有多个选项卡的应用程序,其中一些选项卡涉及过多的计算,而另一些选项卡的计算速度很快。一个允许用户在 react 性或手动更新之间进行选择的复选框,与“刷新”按钮结合使用,将是理想的选择
我知道您可以在获取集合时使用 reactive: false 关闭 react 性。如何在内容可编辑区域内的集合字段中实现相同的效果?示例: Template.documentPage.events(
我想在 z3 中表示一个哈希函数,比如 SHA(x)。在做了一些研究之后,似乎 z3 不能很好地支持注入(inject)性,所以我不能有像这样的约束(虽然我意识到这并不是严格意义上的碰撞,但作为一种启
我正在解决一个问题,我想在仪表板中将数据显示为图表(通过 perak:c3 )和表格(通过 aslagle:reactive-table )。我的问题是数据是从 MongoDB 中的集合中提取的,它的
我的 ViewModel 中有这个函数,它返回一个信号,但内部 block 不起作用,我尝试添加断点,但它没有中断。这是我的代码。 func executeLoginAPI() -> RACSigna
我是一名优秀的程序员,十分优秀!