module ToySolver.Knapsack
( Weight
, Value
, solve
) where
import Control.Monad.State
import Data.Function (on)
import Data.IntSet (IntSet)
import qualified Data.IntSet as IntSet
import Data.List
type Weight = Rational
type Value = Rational
solve
:: [(Value, Weight)]
-> Weight
-> (Value, Weight, [Bool])
solve items limit =
( sum [v | (n,(v,_)) <- zip [0..] items, n `IntSet.member` sol]
, sum [w | (n,(_,w)) <- zip [0..] items, n `IntSet.member` sol]
, [n `IntSet.member` sol | (n,_) <- zip [0..] items]
)
where
items' :: [(Value, Weight, Int)]
items' = map fst $ sortBy (flip compare `on` snd) [((v, w, n), (v / w, v)) | (n, (v, w)) <- zip [0..] items]
sol :: IntSet
sol = IntSet.fromList $ fst $ execState (f items' limit ([],0)) ([],0)
f :: [(Value, Weight, Int)] -> Weight -> ([Int],Value) -> State ([Int],Value) ()
f items !slack (is, !value) = do
(_, bestVal) <- get
when (computeUB items slack value > bestVal) $ do
case items of
[] -> put (is,value)
(v,w,i):items -> do
when (slack >= w) $ f items (slack w) (i : is, v + value)
f items slack (is, value)
computeUB :: [(Value, Weight, Int)] -> Weight -> Value -> Value
computeUB items slack value = go items slack value
where
go _ 0 val = val
go [] _ val = val
go ((v,w,_):items) slack val
| slack >= w = go items (slack w) (val + v)
| otherwise = val + (v / w) * slack