对于如何避免由于可变状态而导致错误的记忆有共识吗?
在此示例中,缓存结果的状态发生了变化,因此在第二次调用时给出了错误的结果。
class Greeter
def initialize
@greeting_cache = {}
end
def expensive_greeting_calculation(formality)
case formality
when :casual then "Hi"
when :formal then "Hello"
end
end
def greeting(formality)
unless @greeting_cache.has_key?(formality)
@greeting_cache[formality] = expensive_greeting_calculation(formality)
end
@greeting_cache[formality]
end
end
def memoization_mutator
greeter = Greeter.new
first_person = "Bob"
# Mildly contrived in this case,
# but you could encounter this in more complex scenarios
puts(greeter.greeting(:casual) << " " << first_person) # => Hi Bob
second_person = "Sue"
puts(greeter.greeting(:casual) << " " << second_person) # => Hi Bob Sue
end
memoization_mutator
我可以看到避免这种情况的方法是:
greeting
可以返回一个dup
或clone
@greeting_cache[formality]
greeting
可能freeze
的结果@greeting_cache[formality]
。memoization_mutator
这会导致在向其附加字符串时引发异常。- 检查所有使用结果的代码,
greeting
以确保它没有对字符串进行任何变异。
是否就最佳方法达成共识?做(1)或(2)的唯一缺点是降低性能吗?(我还怀疑如果对象引用了其他对象,冻结对象可能无法完全工作)
旁注:这个问题不会影响 memoization 的主要应用:由于Fixnum
s 是不可变的,计算斐波那契数列不存在可变状态的问题。:)