Mathematica 中的动态编程:如何自动本地化和/或清除记忆函数的定义
在 Mathematica 8.0 中,假设我有一些常量:
a:=7
b:=9
c:=13
d:=.002
e:=2
f:=1
并且我想用它们来评估一些相互关联的函数
g[0,k_]:=0
g[t_,0]:=e
g[t_,k_]:=g[t-1,k]*a+h[t-1,k-1]*b
h[0,k_]:=0
h[t_,0]:=f
h[t_,k_]:=h[t-1,k]*c+g[t-1,k-1]*d
但这真的很慢并且需要动态编程,否则你会得到指数级的减速:
g[0, k_] := 0
g[t_, 0] := e
g[t_, k_] := g[t, k] = g[t - 1, k]*a + h[t - 1, k - 1]*b
h[0, k_] := 0
h[t_, 0] := f
h[t_, k_] := h[t, k] = h[t - 1, k]*c + g[t - 1, k - 1]*d
现在它真的很快,但如果我们曾经想要更改常量(例如,在 Manipulate 函数中使用它),我们每次都必须Clear
g
和 h
。如果我们有复杂的相互依赖关系,那么每次我们想要从 g
和 h
中获取值时,将它们全部清除可能真的很烦人。
有没有一种简单的方法可以在 Module
或 Block
或类似内容中运行 g
和 h
,以便我可以每次评估时都能得到新的结果,而不会出现指数级减速?或者甚至是一种以良好的方式快速构建 g
和 h
结果表的方法?如前所述,我希望能够在 Manipulate
函数中计算 g
和 h
。
In Mathematica 8.0, suppose I have some constants:
a:=7
b:=9
c:=13
d:=.002
e:=2
f:=1
and I want to use them to evaluate some interlinked functions
g[0,k_]:=0
g[t_,0]:=e
g[t_,k_]:=g[t-1,k]*a+h[t-1,k-1]*b
h[0,k_]:=0
h[t_,0]:=f
h[t_,k_]:=h[t-1,k]*c+g[t-1,k-1]*d
But this is really slow and in need of dynamic programming, or else you get an exponential slowdown:
g[0, k_] := 0
g[t_, 0] := e
g[t_, k_] := g[t, k] = g[t - 1, k]*a + h[t - 1, k - 1]*b
h[0, k_] := 0
h[t_, 0] := f
h[t_, k_] := h[t, k] = h[t - 1, k]*c + g[t - 1, k - 1]*d
Now it's really fast, but if we ever want to change the constants(say, to use this in a Manipulate function) we have to Clear
g
and h
each time. If we had complex interdependencies, it might be really annoying to clear them all out every single time we wanted a value from g
and h
.
Is there an easy way to run g
and h
in a Module
or Block
or similar, so that I can get a fresh result back each time it's evaluated without the exponential slowdown? Or even a fast way to build up a table of results for both g
and h
in a nice way? As said, I want to be able to compute g
and h
in a Manipulate
function.
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
这是一种方法,根据您的要求使用
Block
:我们将使用它来定义
f
和h
,正如我们现在所说的:
有每次调用后不会留下全局记忆定义 -
Block
会在执行退出Block
之前小心地销毁它们。特别是,在这里我将更改参数并再次调用它们:此方案的另一种选择是将所有参数(例如
a,b,c,d,e,f
)显式传递给函数,扩展它们形式参数列表(签名),但这有一个缺点,即与过去不同参数值相对应的旧记忆定义不会被自动清除。这种方法的另一个问题是,由于参数数量的变化等,生成的代码会更加脆弱。编辑
但是,如果您想构建结果表,这可能会更快一些因为您一劳永逸地执行此操作,并且在这种情况下您确实希望保留所有记忆的定义。所以,这是代码:
您调用它,显式传递参数:(
我使用的是原始参数)。您可以在此方法中检查记忆的定义是否保留在全局规则库中。下次您调用具有完全相同参数的函数时,它将获取记忆的定义而不是重新计算。除了我上面概述的这种方法的问题之外,您还应该注意内存使用情况,因为没有任何内容被清除。
Here is one way, using
Block
as you requested:We will use this to give definitions to
f
andh
asWe call now:
There are no global memoized definitions left after each call -
Block
takes care to destroy them just before the execution exitsBlock
. In particular, here I will change the parameters and call them again:An alternative to this scheme would be to explicitly pass all parameters like
a,b,c,d,e,f
to functions, extending their formal parameter lists (signatures), but this has a disadvantage that the older memoized definitions corresponding to different past parameter values would not be automatically cleared. Another problem with that approach is that the resulting code will be more fragile, w.r.t changes in the number of parameters, etc.EDIT
However, if you want to build a table of results, this could be somewhat faster since you do it once and for all, and in this case you do want to keep all memoized definitions. So, here is the code:
You call it, passing the parameters explicitly:
(I was using the original parameters). You can check that the memoized definitions remain in the global rule base, in this method. Next time you call a function with exact same parameters, it will fetch the memoized definition rather than recompute. Apart from the problems with this approach that I outlined above, you should also watch for the memory usage, since nothing gets cleared.
使用辅助符号进行记忆
可以修改问题中展示的记忆技术,以便不需要重新定义
g
和h
。每当需要清除缓存时就建立。这个想法是将记忆值存储在辅助符号上,而不是直接存储在g
和h
上:定义本质上与
g 的原始记忆版本相同
和h
不同之处在于引入了新符号memo
。完成这些定义后,只需使用Clear@memo
即可清除缓存 - 无需重新定义g
和h
。更好的是,可以通过将memo
放置在Block
中来本地化缓存,从而:退出该块时,缓存将被丢弃。
使用建议进行记忆化
原始和修订后的记忆化技术需要对函数
g
和h
进行侵入性更改。有时,事后引入记忆是很方便的。做到这一点的一种方法是使用建议技术——一种类似于 OO 编程中的子类化的函数式编程。 函数建议的特定实现定期出现在StackOverflow 页面。然而,该技术也是侵入性的。让我们考虑另一种技术,在不改变全局定义的情况下向g
和h
添加建议。诀窍是在
Block
中临时重新定义g
和h
。重新定义将首先检查缓存中的结果,如果失败,则从块外部调用原始定义。让我们回到g
和h
的原始定义,它们完全不知道记忆化:该技术的基本架构如下所示:
临时符号
gg<引入 /code> 和
hh
来保存g
和h
的原始定义。然后,g
和h
在本地反弹到新的缓存定义,并根据需要委托给原始定义。以下是 copyDownValues 的定义:为了使这篇文章简短(呃),此“复制”功能仅涉及向下值。一般建议工具还需要考虑上值、下值、符号属性等。
这种通用模式很容易实现自动化,尽管很乏味。下面的宏函数
memoize
可以做到这一点,但几乎没有任何注释:费了一番周折,我们现在可以对
g
的非缓存版本进行外部记忆化,并且h
:将它们放在一起,我们现在可以创建一个响应式
Manipulate
块:LocalizeVariables
和TrackedSymbols
选项是g
和h
对全局符号的依赖关系的产物a
到f
。Memoization Using Auxiliary Symbol
The memoization technique exhibited in the question can be modified so that the definitions of
g
andh
do not need to be re-established whenever the cache needs to be cleared. The idea is to store the memoized values on an auxiliary symbol instead of directly ong
andh
:The definitions are essentially the same as the original memoized versions of
g
andh
except that a new symbol,memo
, has been introduced. With these definitions in place, the cache can be cleared simply usingClear@memo
-- there is no need to redefineg
andh
anew. Better still, the cache can be localized by placingmemo
inBlock
, thus:The cache is discarded when the block is exited.
Memoization Using Advice
The original and revised memoization techniques required invasive changes within the function
g
andh
. Sometimes, it is convenient to introduce memoization after the fact. One way to do this would be to use the technique of advising -- a kind of functional programming analog to subclassing in OO programming. A particular implementation of function advice appears regularly in the pages of StackOverflow. However, that technique is also invasive. Let's consider an alternate technique of adding advice tog
andh
without altering their global definitions.The trick will be to temporarily redefine
g
andh
within aBlock
. The redefinitions will first check the cache for the result and, failing that, call the original definitions from outside the block. Let's go back to the original definitions ofg
andh
that are blissfully unaware of memoization:The basic schema for this technique looks like this:
The temporary symbols
gg
andhh
are introduced to hold the original definitions ofg
andh
. Theng
andh
are locally rebound to new caching definitions that delegate to the original definitions as necessary. Here is definition ofcopyDownValues
:In an effort to keep this post short(er), this "copy" function is concerned only with down-values. A general advice facility also needs to account for up-values, subvalues, symbol attributes and so on.
This general pattern is easy, if tedious, to automate. The following macro function
memoize
does this, presented with little comment:After much ado, we are now in a position to externally impose memoization upon the non-caching versions of
g
andh
:Putting it all together, we can now create a responsive
Manipulate
block:The
LocalizeVariables
andTrackedSymbols
options are artifacts of the dependencies thatg
andh
have on the global symbolsa
throughf
.