Skip to content

Commit d60d28b

Browse files
committed
make toUnfoldable ordered
addresses purescript-deprecated#135 toAscUnfoldable becomes a deprecated equivalent to toUnfoldable. We introduce `unsafeToUnfoldable` for the unsorted traversal.
1 parent e424494 commit d60d28b

File tree

2 files changed

+31
-17
lines changed

2 files changed

+31
-17
lines changed

src/Data/Map.purs

+28-13
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ module Data.Map
2121
, fromFoldable
2222
, fromFoldableWith
2323
, toUnfoldable
24+
, unsafeToUnfoldable
2425
, toAscUnfoldable
2526
, delete
2627
, pop
@@ -65,7 +66,7 @@ data Map k v
6566

6667
-- Internal use
6768
toAscArray :: forall k v. Map k v -> Array (Tuple k v)
68-
toAscArray = toAscUnfoldable
69+
toAscArray = toUnfoldable
6970

7071
instance eq1Map :: Eq k => Eq1 (Map k) where
7172
eq1 = eq
@@ -554,20 +555,9 @@ fromFoldableWith f = foldl (\m (Tuple k v) -> alter (combine v) k m) empty where
554555
combine v (Just v') = Just $ f v v'
555556
combine v Nothing = Just v
556557

557-
-- | Convert a map to an unfoldable structure of key/value pairs
558+
-- | Convert a map to an unfoldable structure of key/value pairs where the keys are in ascending order
558559
toUnfoldable :: forall f k v. Unfoldable f => Map k v -> f (Tuple k v)
559560
toUnfoldable m = unfoldr go (m : Nil) where
560-
go Nil = Nothing
561-
go (hd : tl) = case hd of
562-
Leaf -> go tl
563-
Two left k v right ->
564-
Just $ Tuple (Tuple k v) (left : right : tl)
565-
Three left k1 v1 mid k2 v2 right ->
566-
Just $ Tuple (Tuple k1 v1) (singleton k2 v2 : left : mid : right : tl)
567-
568-
-- | Convert a map to an unfoldable structure of key/value pairs where the keys are in ascending order
569-
toAscUnfoldable :: forall f k v. Unfoldable f => Map k v -> f (Tuple k v)
570-
toAscUnfoldable m = unfoldr go (m : Nil) where
571561
go Nil = Nothing
572562
go (hd : tl) = case hd of
573563
Leaf -> go tl
@@ -580,6 +570,31 @@ toAscUnfoldable m = unfoldr go (m : Nil) where
580570
Three left k1 v1 mid k2 v2 right ->
581571
go $ left : singleton k1 v1 : mid : singleton k2 v2 : right : tl
582572

573+
-- | Convert a map to an unfoldable structure of key/value pairs
574+
--
575+
-- While this traversal is up to 10% faster in benchmarks than `toUnfoldable`,
576+
-- it leaks the underlying map stucture, making it only suitable for applications
577+
-- where order is irrelevant.
578+
--
579+
-- If you are unsure, use `toUnfoldable`
580+
unsafeToUnfoldable :: forall f k v. Unfoldable f => Map k v -> f (Tuple k v)
581+
unsafeToUnfoldable m = unfoldr go (m : Nil) where
582+
go Nil = Nothing
583+
go (hd : tl) = case hd of
584+
Leaf -> go tl
585+
Two left k v right ->
586+
Just $ Tuple (Tuple k v) (left : right : tl)
587+
Three left k1 v1 mid k2 v2 right ->
588+
Just $ Tuple (Tuple k1 v1) (singleton k2 v2 : left : mid : right : tl)
589+
590+
-- | Deprecated in favor of `toUnfoldable,` which now also has ascending key order.
591+
toAscUnfoldable
592+
:: forall f k v
593+
. Warn "Map.toAscUnfoldable is deprecated. Use toUnfoldable."
594+
=> Unfoldable f
595+
=> Map k v -> f (Tuple k v)
596+
toAscUnfoldable = toUnfoldable
597+
583598
-- | Get a list of the keys contained in a map
584599
keys :: forall k v. Map k v -> List k
585600
keys Leaf = Nil

test/Test/Data/Map.purs

+3-4
Original file line numberDiff line numberDiff line change
@@ -175,11 +175,10 @@ mapTests = do
175175
groupBy ((==) `on` fst) <<< sortBy (compare `on` fst) in
176176
M.fromFoldableWith (<>) arr === f (arr :: List (Tuple String String))
177177

178-
log "toAscUnfoldable is sorted version of toUnfoldable"
178+
log "toUnfoldable is sorted"
179179
quickCheck $ \(TestMap m) ->
180180
let list = M.toUnfoldable (m :: M.Map SmallKey Int)
181-
ascList = M.toAscUnfoldable m
182-
in ascList === sortBy (compare `on` fst) list
181+
in list === sortBy (compare `on` fst) list
183182

184183
log "Lookup from union"
185184
quickCheck $ \(TestMap m1) (TestMap m2) k ->
@@ -283,7 +282,7 @@ mapTests = do
283282

284283
log "filterWithKey keeps those keys for which predicate is true"
285284
quickCheck $ \(TestMap s :: TestMap String Int) p ->
286-
A.all (uncurry p) (M.toAscUnfoldable (M.filterWithKey p s) :: Array (Tuple String Int))
285+
A.all (uncurry p) (M.toUnfoldable (M.filterWithKey p s) :: Array (Tuple String Int))
287286

288287
log "filterKeys gives submap"
289288
quickCheck $ \(TestMap s :: TestMap String Int) p ->

0 commit comments

Comments
 (0)