- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我经常读到惰性与非严格不同,但我发现很难理解其中的区别。它们似乎可以互换使用,但我知道它们有不同的含义。我希望能得到一些帮助来理解其中的差异。
我有一些关于这篇文章的问题。我将在本文末尾总结这些问题。我有一些示例片段,我没有测试它们,我只是将它们作为概念呈现。我添加了引号,以免您查找它们。也许它会对以后遇到同样问题的人有所帮助。
A function f is said to be strict if, when applied to a nonterminatingexpression, it also fails to terminate. In other words, f is strictiff the value of f bot is |. For most programming languages, allfunctions are strict. But this is not so in Haskell. As a simpleexample, consider const1, the constant 1 function, defined by:
const1 x = 1
The value of const1 bot in Haskell is 1. Operationally speaking, sinceconst1 does not "need" the value of its argument, it never attempts toevaluate it, and thus never gets caught in a nonterminatingcomputation. For this reason, non-strict functions are also called"lazy functions", and are said to evaluate their arguments "lazily",or "by need".
-A Gentle Introduction To Haskell: Functions
我真的很喜欢这个定义。这似乎是我能找到的最好的理解严格的方法。 const1 x = 1
也是懒惰的吗?
Non-strictness means that reduction (the mathematical term forevaluation) proceeds from the outside in,
so if you have (a+(bc)) then first you reduce the +, then you reducethe inner (bc).
-Haskell Wiki: Lazy vs non-strict
Haskell Wiki 真的让我很困惑。我理解他们所说的关于顺序的内容,但我不明白如果通过 _|_
,(a+(b*c))
将如何非严格地评估?
In non-strict evaluation, arguments to a function are not evaluatedunless they are actually used in the evaluation of the function body.
Under Church encoding, lazy evaluation of operators maps to non-strictevaluation of functions; for this reason, non-strict evaluation isoften referred to as "lazy". Boolean expressions in many languages usea form of non-strict evaluation called short-circuit evaluation, whereevaluation returns as soon as it can be determined that an unambiguousBoolean will result — for example, in a disjunctive expression wheretrue is encountered, or in a conjunctive expression where false isencountered, and so forth. Conditional expressions also usually uselazy evaluation, where evaluation returns as soon as an unambiguousbranch will result.
-Wikipedia: Evaluation Strategy
Lazy evaluation, on the other hand, means only evaluating anexpression when its results are needed (note the shift from"reduction" to "evaluation"). So when the evaluation engine sees anexpression it builds a thunk data structure containing whatever valuesare needed to evaluate the expression, plus a pointer to theexpression itself. When the result is actually needed the evaluationengine calls the expression and then replaces the thunk with theresult for future reference....
Obviously there is a strong correspondence between a thunk and apartly-evaluated expression. Hence in most cases the terms "lazy" and"non-strict" are synonyms. But not quite.
-Haskell Wiki: Lazy vs non-strict
这似乎是 Haskell 的具体答案。我认为惰性意味着thunk,非严格意味着部分评估。这样的比较是不是太简单化了? lazy 是否总是意味着 thunk,而 non-strict 总是意味着部分评估。
In programming language theory, lazy evaluation or call-by-need1 isan evaluation strategy which delays the evaluation of an expressionuntil its value is actually required (non-strict evaluation) and alsoavoid repeated evaluations (sharing).
我知道大多数人都会说在学习函数式语言时忘记命令式编程。但是,我想知道这些是否符合非严格、惰性、两者兼有或两者都没有的资格?至少它会提供一些熟悉的东西。
短路
f1() || f2()
C#、Python 等具有“yield”的语言
public static IEnumerable Power(int number, int exponent)
{
int counter = 0;
int result = 1;
while (counter++ < exponent)
{
result = result * number;
yield return result;
}
}
回调
int f1() { return 1;}
int f2() { return 2;}
int lazy(int (*cb1)(), int (*cb2)() , int x) {
if (x == 0)
return cb1();
else
return cb2();
}
int eager(int e1, int e2, int x) {
if (x == 0)
return e1;
else
return e2;
}
lazy(f1, f2, x);
eager(f1(), f2(), x);
我知道答案就在我面前,拥有所有这些资源,但我无法捕获它。看起来这个定义太容易被认为是隐含的或明显的而被忽视。
我知道我有很多问题。请随意回答您认为相关的任何问题。我添加了这些问题以供讨论。
const1 x = 1
也是懒惰的吗?非常感谢!
最佳答案
非严格和惰性虽然非正式地可以互换,但适用于不同的讨论领域。
非严格指的是 semantics :表达式的数学含义。非严格适用的世界没有函数的运行时间、内存消耗甚至计算机的概念。它只是讨论域中的哪些类型的值映射到共域中的哪些类型的值。特别是,严格函数必须将值 ⊥(“bottom”——有关更多信息,请参阅上面的语义链接)映射到 ⊥;允许非严格函数不执行此操作。
惰性指的是操作行为:代码在真实计算机上执行的方式。大多数程序员从操作角度考虑程序,所以这可能就是您的想法。惰性求值是指使用 thunk 的实现——指向代码的指针,这些代码在第一次执行时被值替换。注意这里的非语义词:“指针”、“第一次”、“执行”。
惰性求值会产生不严格的语义,这就是为什么这些概念看起来如此接近的原因。但正如 FUZxxl 指出的那样,惰性并不是实现非严格语义的唯一方法。
如果您有兴趣了解有关此区别的更多信息,我强烈推荐上面的链接。阅读这本书是我对计算机程序含义概念的转折点。
关于haskell - 非严格和惰性有何不同?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7140978/
我先说我正在学习 Haskel,所以不要太苛刻。 Haskell 的惰性求值可能有用也可能危险,这取决于计算的瓶颈是时间复杂度还是堆栈的大小。 出于这个原因,我想更好地了解 Haskell 中求值的工
我正在开发一款玩具 RTS 游戏,我依赖 A* 寻找路径,问题是很多单位四处移动导致计算的路径变得无效,这导致 CPU 周期浪费,我必须为那些重新计算路径代理商。 所以我想为什么不懒惰地计算路径而不是
我正在尝试非贪婪地解析出 TD 标签。我从这样的事情开始: stuffMore stuffOther stuffthingsmore things 我使用以下作为我的正则表达式: Regex.Spli
我正在学习 http://learnyouahaskell.com/starting-out 上的(优秀的)Haskell 教程。并且正在尝试直角三角形示例: > let triangles = [(
我编写了一个小型 Haskell 程序来打印当前目录中所有文件的 MD5 校验和(递归搜索)。基本上是 md5deep 的 Haskell 版本.一切都很好,除非当前目录有大量文件,在这种情况下我会收
我通常听说生产代码应该避免使用惰性 I/O。我的问题是,为什么?除了闲逛之外,还可以使用 Lazy I/O 吗?是什么让替代方案(例如枚举器)更好? 最佳答案 惰性 IO 存在的问题是,释放您所获取的
我注意到 Scala 提供了lazy vals。但我不明白他们在做什么。 scala> val x = 15 x: Int = 15 scala> lazy val y = 13 y: Int =
我目前正在尝试将 XML 文件的内容读入 Map Int (Map Int String) 并且它工作得很好(使用 HaXml)。但是,我对程序的内存消耗不满意,问题似乎出在垃圾回收上。 这是我用来读
lazy val seq: Unit = { println("a") seq } 我们可以尾递归调用上面的表达式吗? 最佳答案 我想你可以从这个意义上说,是的 - 评估时,seq将递归评估自
在以下示例中: def maybeTwice2(b: Boolean, i: => Int) = { lazy val j = i if (b) j+j else 0 } 为什么当我这样调用它
我的一个项目使用了混合的Scala功能,这些功能似乎不能很好地融合在一起: 类型类和无形自动类型类实例派生 隐式转换(向具有类型类实例的类型添加有用的语法) 默认参数,因为即使它们通常是一件坏事,但在
我有一个应用程序,涉及一个数组集合,这些数组可能非常大(索引最大为 int 的最大值),但它们是惰性 - 它们内容是动态计算的,并且在请求之前实际上是不知道的。数组也是不可变的 - 每个数组的每个元素
最近我开始使用 spring 中的惰性初始化功能很多。所以我一直在徘徊——懒惰地初始化你的 bean 有什么实际的缺点吗?如果不是 - 为什么不是懒惰的默认行为? 最佳答案 主要的“缺点”是不能立即发
我有一个通过信息亭向访问者显示的网站。人们可以与之互动。但是,由于该网站不是本地托管的,而是使用互联网连接 - 页面加载速度很慢。 我想实现某种惰性缓存机制,以便在人们浏览页面时 - 页面和页面引用的
我是否正确理解声明关系急切加载的方法是使用lazy='joined'或lazy='subquery'? “lazy”与“eager”相反——在这种情况下使用“lazy”关键字来表示急切加载,这是一个历
我想抓取 对之间任何值的内容标签。 This is one block of text This is another one 我想出的正则表达式是 /(.*)/m 虽然,它看起来很贪心,并
考虑以下几点: z = [{"x" => 5}, 2, 3].lazy.map{ |i| i} #=> #5}, 2, 3]>:map> z.first #=> {"x"=>5} 当我尝试将 z 转换
因此我有一个条件语句: if($boolean && expensiveOperation()){ ...} PHP 是否具有惰性 bool 值评估,即它是否会检查 $boolean 并且如果它为 f
我在 Scala 上有问题。我用 @transient lazy val 序列化了一个类的实例 field 。然后我反序列化它,该字段被分配null .我期待反序列化后的惰性评估。我该怎么办? 以下是
我编写了以下函数,我认为该函数应该以原子方式执行 IO(只要其他人都使用相同的 MVar)。 atomicIO :: MVar () -> IO a -> IO a atomicIO mvar io
我是一名优秀的程序员,十分优秀!