- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
以下是蛮力毕达哥拉斯三胞胎问题的三个版本,其附加约束条件为a + b + c = 1000。所有这些均符合GHC 7.0.3的-O3标准。示例运行时在下面列出。
问题:
main=print . product . head $ [[a,b,c] | a<-[1..1000],b<-[a..1000], let c=1000-a-b, a^2+b^2==c^2]
real 0m0.046s
user 0m0.039s
sys 0m0.005s
main=print . product . head $ [[a,b,c] | a<-[1..1000],b<-[1..1000], let c=1000-a-b, a^2+b^2==c^2]
real 0m0.045s
user 0m0.036s
sys 0m0.006s
main=print . product . head $ [[a,b,c] | a<-[1..1000],b<-[1..1000], b>=a, let c=1000-a-b, a^2+b^2==c^2]
real 0m0.040s
user 0m0.033s
sys 0m0.005s
最佳答案
让我们分别命名这三个程序 A , B 和 C 。
C 与 B
我们将从最简单的开始: C 相对于 B 有一个额外的约束(b >= a
)。从直觉上讲,这意味着在 C 中遍历的搜索空间小于 B 的搜索空间。我们可以合理地指出这一点,而不是对所有可能的偶对a, b
(我们知道其中有1000^2=1000000
个对)进行覆盖,而不是考虑b
小于a
的所有情况。大概是检查b >= a
是否产生一点额外的代码(比较),而运行该比较避免了计算所产生的额外代码(比较),因此,我们注意到(略微)的加速。很公平。
B 和 A
接下来的问题有些棘手:和 C (b >= a
)具有相同的约束条件,但是编码方式不同(即在这里我们将其编码为b
中可能达到的值List Monad
的范围)。那时我们可能会认为 A 的运行速度应比 B 的运行速度更快(事实上,它的运行方式应类似于 C )。显然,我们缺乏直觉。
正中核心
现在,由于我们不能一直相信自己的直觉,因此我们应该研究生成的GHC Core中实际发生的情况。让我们转储我们的3个程序的核心(无优化):
for p in A B C
do
ghc -ddump-simpl $p.hs >> $p.core
done
B.core
和C.core
,我们会注意到两个文件的结构大致相同:(System.IO.print ...)
,(Data.List.product ...)
和(GHC.List.head ...)
ds_dxd [Occ=LoopBreaker]
:: [GHC.Integer.Type.Integer] -> [[GHC.Integer.Type.Integer]]
(GHC.Enum.enumFromTo
@ GHC.Integer.Type.Integer
GHC.Num.$fEnumInteger
(GHC.Integer.smallInteger 1)
(GHC.Integer.smallInteger 1000)))
B.core
中看到 case GHC.Classes.==
@ GHC.Integer.Type.Integer
...
(GHC.Num.+
...
(GHC.Real.^
...
ds3_dxc
(GHC.Integer.smallInteger 2))
(GHC.Real.^
...
ds8_dxg
(GHC.Integer.smallInteger 2)))
(GHC.Real.^
...
c_abw
(GHC.Integer.smallInteger 2))
C.core
中,我们改为:case GHC.Classes.>=
@ GHC.Integer.Type.Integer GHC.Classes.$fOrdInteger ds8_dxj ds3_dxf
of _ {
GHC.Bool.False -> ds5_dxh ds9_dxk;
GHC.Bool.True ->
let {
...
case GHC.Classes.==
...
>=
约束,因此,正如我们的直觉所期望的那样,搜索更少的整数以缩短运行时间。A.core
和B.core
时,我们立即看到一个熟悉的结构(一对嵌套的递归函数,每个都通过枚举调用)并且实际上,看来A
和B
的核心输出几乎相同!区别似乎在于最内层的枚举: ds5_dxd
(GHC.Enum.enumFromTo
@ GHC.Integer.Type.Integer
GHC.Num.$fEnumInteger
ds3_dxb
(GHC.Integer.smallInteger 1000))
ds3_dxb
到1000
的枚举范围,而不是保留为静态范围([1..1000
])。ghc
和gcc
。为了有机会了解这些结果,我们必须依赖于生成的优化内核(尽管最终,真正重要的是生成的汇编器,但现在我们将忽略它)。for p in A B C
do
ghc -O3 -ddump-simpl $p.hs >> $p.core
done
40
的代码行,从而看到更紧密的内部循环,从而看到这一点。要了解为什么无法从此优化中受益,我们应该看一下没有浮出水面的代码:let {
c1_s10T
:: GHC.Integer.Type.Integer
-> [[GHC.Integer.Type.Integer]]
-> [[GHC.Integer.Type.Integer]]
[LclId, Arity=2, Str=DmdType LL]
c1_s10T =
\ (ds2_dxg :: GHC.Integer.Type.Integer)
(ds3_dxf :: [[GHC.Integer.Type.Integer]]) ->
let {
c2_s10Q [Dmd=Just L] :: GHC.Integer.Type.Integer
[LclId, Str=DmdType]
c2_s10Q = GHC.Integer.minusInteger lvl2_s10O ds2_dxg } in -- subtract
case GHC.Integer.eqInteger
(GHC.Integer.plusInteger lvl3_s10M (GHC.Real.^_^ ds2_dxg lvl_r11p))
-- add two squares (lve3_s10M has been floated out)
(GHC.Real.^_^ c2_s10Q lvl_r11p)
-- ^ compared to this square
of _ {
GHC.Bool.False -> ds3_dxf;
GHC.Bool.True ->
GHC.Types.:
@ [GHC.Integer.Type.Integer]
(GHC.Types.:
@ GHC.Integer.Type.Integer
ds_dxe
(GHC.Types.:
@ GHC.Integer.Type.Integer
ds2_dxg
(GHC.Types.:
@ GHC.Integer.Type.Integer
c2_s10Q
(GHC.Types.[] @ GHC.Integer.Type.Integer))))
ds3_dxf
} } in
minusInteger
)和等式(eqInteger
)以及两个平方(^_^
),而在C.core
中执行相同的辅助函数包含较少的计算(如果进一步研究,我们会发现这是因为GHC无法确定是否可以安全地在优化过程中进行这些计算)。这与我们前面的分析相吻合,因为我们可以看到约束(b >= a
)实际上存在,与 C 不同,我们无法将大部分冗余计算浮到循环外。[1..10000]
。我们应该期望看到 A的运行时行为应渐近于 C的运行时行为,就像我们期望 B 一样。➜ time ./A
./A 0.37s user 0.01s system 74% cpu 0.502 total
➜ time ./B
./B 3.21s user 0.02s system 99% cpu 3.246 total
➜ time ./C
./C 0.33s user 0.01s system 99% cpu 0.343 total
关于performance - haskell列表理解能力,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6454370/
在 Haskell 中,类型声明使用双冒号,即 (::),如 not::Bool -> Bool。 但是在许多语法与 Haskell 类似的语言中,例如榆树、 Agda 、他们使用单个冒号(:)来声明
insertST :: StateDecoder -> SomeState -> Update SomeState SomeThing insertST stDecoder st = ... Stat
如果这个问题有点含糊,请提前道歉。这是一些周末白日梦的结果。 借助 Haskell 出色的类型系统,将数学(尤其是代数)结构表达为类型类是非常令人愉快的。我的意思是,看看 numeric-prelud
我有需要每 5 分钟执行一次的小程序。 目前,我有执行该任务的 shell 脚本,但我想通过 CLI 中的键为用户提供无需其他脚本即可运行它的能力。 实现这一目标的最佳方法是什么? 最佳答案 我想你会
RWH 面世已经有一段时间了(将近 3 年)。在在线跟踪这本书的渐进式写作之后,我渴望获得我的副本(我认为这是写书的最佳方式之一。)在所有相当学术性的论文中,作为一个 haskell 学生,读起来多么
一个经典的编程练习是用 Lisp/Scheme 编写一个 Lisp/Scheme 解释器。可以利用完整语言的力量来为该语言的子集生成解释器。 Haskell 有类似的练习吗?我想使用 Haskell
以下摘自' Learn You a Haskell ' 表示 f 在函数中用作“值的类型”。 这是什么意思?即“值的类型”是什么意思? Int 是“值的类型”,对吗?但是 Maybe 不是“值的类型”
现在我正在尝试创建一个基本函数,用于删除句子中的所有空格或逗号。 stringToIntList :: [Char] -> [Char] stringToIntList inpt = [ a | a
我是 Haskell 的新手,对模式匹配有疑问。这是代码的高度简化版本: data Value = MyBool Bool | MyInt Integer codeDuplicate1 :: Valu
如何解释这个表达式? :t (+) (+3) (*100) 自 和 具有相同的优先级并且是左结合的。我认为这与 ((+) (+3)) (*100) 相同.但是,我不知道它的作用。在 Learn
这怎么行 > (* 30) 4 120 但这不是 > * 30 40 error: parse error on input ‘*’ 最佳答案 (* 30) 是一个 section,它仍然将 * 视为
我想创建一个函数,删除满足第二个参数中给定谓词的第一个元素。像这样: removeFirst "abab" ( 'b') = "abab" removeFirst [1,2,3,4] even =
Context : def fib(n): if n aand returns a memoized version of the same function. The trick is t
我明白惰性求值是什么,它是如何工作的以及它有什么优势,但是你能解释一下 Haskell 中什么是严格求值吗?我似乎找不到太多关于它的信息,因为惰性评估是最著名的。 他们各自的优势是什么。什么时候真正使
digits :: Int -> [Int] digits n = reverse (x) where x | n digits 1234 = [3,1,2,4]
我在 F# 中有以下代码(来自一本书) open System.Collections.Generic type Table = abstract Item : 'T -> 'U with ge
我对 Haskell 比较陌生,过去几周一直在尝试学习它,但一直停留在过滤器和谓词上,我希望能得到帮助以帮助理解。 我遇到了一个问题,我有一个元组列表。每个元组包含一个 (songName, song
我是 haskell 的初学者,我试图为埃拉托色尼筛法定义一个简单的函数,但它说错误: • Couldn't match expected type ‘Bool -> Bool’
我是 Haskell 语言的新手,我在使用 read 函数时遇到了一些问题。准确地说,我的理解是: read "8.2" + 3.8 应该返回 12.0,因为我们希望返回与第二个成员相同的类型。我真正
当我尝试使用真实项目来驱动它来学习 Haskell 时,我遇到了以下定义。我不明白每个参数前面的感叹号是什么意思,我的书上好像也没有提到。 data MidiMessage = MidiMessage
我是一名优秀的程序员,十分优秀!