3 回答
TA贡献1911条经验 获得超7个赞
语法[e1, e2 .. e3]实际上是语法糖enumFromThenTo e1 e2 e3,它是Enum类型类中的一个函数。
Haskell标准定义了它的语义如下:
对于类型Int和Integer,枚举函数具有以下含义:
序列enumFrom e1是列表[e1,e1 + 1,e1 + 2,…]。
序列enumFromThen e1 e2是列表[e1,e1 + i,e1 + 2i,…],其中增量i为e2 − e1。增量可以是零或负数。如果增量为零,则所有列表元素都相同。
序列enumFromTo e1 e3是列表[e1,e1 + 1,e1 + 2,…e3]。如果列表是空的e1 > e3。
序列enumFromThenTo e1 e2 e3是列表[e1,e1 + i,e1 +
2i,…e3],其中增量i为e2 − e1。如果增量为正或零,则列表在下一个元素大于时终止e3; 如果列表是空的e1 > e3。如果增量为负,则列表在下一个元素小于时终止 e3; 如果列表是空的e1 < e3。
这几乎是您所期望的,但是Float和Double实例的定义不同:
对于Float和Double,系列的语义enumFrom由Int上面的规则给出,除了当元素变得大于e3 + i∕2正增量时i,或者当它们变得小于e3 + i∕2负数时,列表终止i。
我不确定这是什么理由,所以我能给你的唯一答案就是这样,因为它在标准中就是这样定义的。
您可以通过使用整数枚举并转换为之Float后来解决此问题。
Prelude> map fromIntegral [1, 3 .. 10] :: [Float]
[1.0,3.0,5.0,7.0,9.0]
TA贡献1824条经验 获得超8个赞
这样的总和通常也会在省略最后一项时给出错误的结果; 只有在特殊情况下sqrt (1 - 1)
才会这样。我再次争辩说明显错误的 NaN比一个听起来不错的订单0不准确的浮点数更不好。 - 要确定这样的东西,使用可能触及域边界的浮动只是从根本上说是错误的方法; 一个应该计算单元格中点,或者仅使用范围语法用于单元间边界,但是分别添加精确的外部边界。
添加回答
举报