計算機科学のブログ

コンテキストでの型の操作 Applicative型クラス:関数をコンテキスト内で使用する 2つの都市の距離を計算するコマンドラインアプリケーション Functorの制限

入門Haskellプログラミング (Will Kurt(著)、株式会社クイープ(監修、翻訳)、翔泳社)のUNIT5(コンテキストでの型の操作)、LESSON 28(Applicative型クラス:関数をコンテキスト内で使用する)、28.1(2つの都市の距離を計算するコマンドラインアプリケーション)、Functorの制限、クイックチェック 28-2の解答を求めてみる。

コード

lesson/app/Main.hs

module Main where

import Lib
  ( distanceFromNY,
  )

main :: IO ()
main = do
  mapM_
    (print . distanceFromNY)
    [ (42.6054, -70.7829),
      (42.8250, -70.8150)
    ]

lesson/src/Lib.hs

module Lib
  ( distanceFromNY,
  )
where

type LatLong = (Double, Double)

toRadians :: Floating a => a -> a
toRadians degrees = degrees * pi / 180

latLongToRads :: LatLong -> (Double, Double)
latLongToRads (lat, long) = (toRadians lat, toRadians long)

haversine :: LatLong -> LatLong -> Double
haversine coords1 coords2 =
  let (rlat1, rlong1) = latLongToRads coords1
      (rlat2, rlong2) = latLongToRads coords2
      dlat = rlat2 - rlat1
      dlong = rlong2 - rlong1
      a = sin (dlat / 2) ^ 2 + cos rlat1 * cos rlat2 * sin (dlong / 2) ^ 2
      c = 2 * atan2 (sqrt a) (sqrt (1 - a))
      earthRadius = 3961.0
   in earthRadius * c

distanceFromNY :: LatLong -> Double
distanceFromNY = haversine (40.7776, -73.9691)

入出力結果(Terminal, Zsh)

% stack exec lesson-exe
207.3909006336738
215.50763229827754
%