微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

Haskell 类型签名 pi

如何解决Haskell 类型签名 pi

我对 Haskell 和函数式编程非常陌生。

我想用一个程序来近似 pi。我的代码有效,但只有没有类型签名。 然而任务是编写一个适合给定类型签名的列表理解,所以我可能不得不改变我的列表理解。

pi_approx :: Int -> Double

pi_approx n = sqrt (6 * (sum [1 / x^2 | x <- [1..n]])) 

解决方法

这对签名 Int -> Double 不起作用的原因是因为 (*) :: Num a => a -> a -> asqrt :: Floating a => a -> a 之类的函数要求操作数和结果都具有相同的类型。

这意味着如果 n 是一个 Int,那么 x 中的 x <- [1..n] 也是一个 Int,并且由于 {{ 1}} 是 (^) :: (^) :: (Integral b,Num a) => a -> b -> a,这意味着 (^)x ^ 2 具有相同的类型,因此这也是一个 x

现在出现的问题是 (/) :: Fractional a => a -> a -> a 需要类型为 Int 类型类的成员,而 Fractional 不是 Int

我们可以使用 fromIntegral :: (Integral a,Num b) => a -> b 将值从 Fractional 类型类的成员类型转换为 Integral 类型类的成员类型的值。

因此我们可以通过以下方式解决这个问题:

Num

如果 pi_approx :: Int -> Double pi_approx n = sqrt (6 * (sum [1 / fromIntegral (x^2) | x <- [1..n]])) 可以很大,作为 @dfeuer says,在计算平方之前将 x 转换为 double 是有意义的:

x

这会产生:

pi_approx :: Int -> Double
pi_approx n = sqrt (6 * (sum [1 / fromIntegral x^2 | x <- [1..n]]))

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。