Skip to content

Instantly share code, notes, and snippets.

@delirehberi
Last active February 22, 2020 01:40
Show Gist options
  • Save delirehberi/0b0fb21dc197cdf68ef8ffbfb5f958fa to your computer and use it in GitHub Desktop.
Save delirehberi/0b0fb21dc197cdf68ef8ffbfb5f958fa to your computer and use it in GitHub Desktop.
shortest_way.hs
import Data.List
main = do
contents <- getContents
let
threes = groupsOf 3 (map read $ lines contents)
roadSystem = map (\[a,b,c] -> Section a b c) threes
path = optimalPath roadSystem
pathString = concat $ map (show . fst) path
pathPrice = sum $ map snd path
putStrLn $ "The best path to take is: " ++ pathString
putStrLn $ "The price is: " ++ show pathPrice
data Node = Node Road (Maybe Road)
data Road = Road Int Node
data Section = Section {getA::Int, getB::Int, getC::Int} deriving (Show)
type RoadSystem = [Section]
heathrowToLondon :: RoadSystem
heathrowToLondon = [Section 50 10 30, Section 5 90 20, Section 40 2 25, Section 10 8 0]
data Label = A | B | C deriving (Show)
type Path = [(Label, Int)]
roadStep:: (Path,Path) -> Section -> (Path, Path)
roadStep (pathA, pathB) (Section a b c) =
let
priceA = sum $ map snd pathA
priceB = sum $ map snd pathB
forwardPriceToA = priceA + a
crossPriceToA = priceB + b+ c
forwardPriceToB = priceB + b
crossPriceToB = priceA + a+ c
newPathToA = if forwardPriceToA <= crossPriceToA
then (A,a):pathA
else (C,c):(B,b):pathB
newPathToB = if forwardPriceToB <= crossPriceToB
then (B,b):pathB
else (C,c):(A,a):pathA
in (newPathToA, newPathToB)
optimalPath :: RoadSystem -> Path
optimalPath roadSystem =
let
(bestAPath, bestBPath) = foldl roadStep ([],[]) roadSystem
in
if sum (map snd bestAPath) <= sum (map snd bestBPath)
then reverse bestAPath
else reverse bestBPath
groupsOf :: Int -> [a] -> [[a]]
groupsOf 0 _ = undefined
groupsOf _ [] = []
groupsOf n xs = take n xs: groupsOf n (drop n xs)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment