Skip to content

Commit 484181f

Browse files
committed
makes toUnfoldable ordered
1 parent 53ad197 commit 484181f

File tree

3 files changed

+24
-18
lines changed

3 files changed

+24
-18
lines changed

src/Data/Map.purs

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ module Data.Map
55

66
import Prelude
77

8-
import Data.Map.Internal (Map, alter, checkValid, delete, empty, filter, filterKeys, filterWithKey, findMax, findMin, foldSubmap, fromFoldable, fromFoldableWith, insert, isEmpty, isSubmap, lookup, lookupGE, lookupGT, lookupLE, lookupLT, member, pop, showTree, singleton, size, submap, toAscUnfoldable, toUnfoldable, union, unionWith, unions, update, values)
8+
import Data.Map.Internal (Map, alter, checkValid, delete, empty, filter, filterKeys, filterWithKey, findMax, findMin, foldSubmap, fromFoldable, fromFoldableWith, insert, isEmpty, isSubmap, lookup, lookupGE, lookupGT, lookupLE, lookupLT, member, pop, showTree, singleton, size, submap, toUnfoldable, toUnfoldableUnordered, union, unionWith, unions, update, values)
99
import Data.Set (Set)
1010
import Unsafe.Coerce (unsafeCoerce)
1111

src/Data/Map/Internal.purs

+20-14
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ module Data.Map.Internal
2121
, fromFoldable
2222
, fromFoldableWith
2323
, toUnfoldable
24-
, toAscUnfoldable
24+
, toUnfoldableUnordered
2525
, delete
2626
, pop
2727
, member
@@ -63,7 +63,7 @@ data Map k v
6363

6464
-- Internal use
6565
toAscArray :: forall k v. Map k v -> Array (Tuple k v)
66-
toAscArray = toAscUnfoldable
66+
toAscArray = toUnfoldable
6767

6868
instance eq1Map :: Eq k => Eq1 (Map k) where
6969
eq1 = eq
@@ -552,20 +552,9 @@ fromFoldableWith f = foldl (\m (Tuple k v) -> alter (combine v) k m) empty where
552552
combine v (Just v') = Just $ f v v'
553553
combine v Nothing = Just v
554554

555-
-- | Convert a map to an unfoldable structure of key/value pairs
555+
-- | Convert a map to an unfoldable structure of key/value pairs where the keys are in ascending order
556556
toUnfoldable :: forall f k v. Unfoldable f => Map k v -> f (Tuple k v)
557557
toUnfoldable m = unfoldr go (m : Nil) where
558-
go Nil = Nothing
559-
go (hd : tl) = case hd of
560-
Leaf -> go tl
561-
Two left k v right ->
562-
Just $ Tuple (Tuple k v) (left : right : tl)
563-
Three left k1 v1 mid k2 v2 right ->
564-
Just $ Tuple (Tuple k1 v1) (singleton k2 v2 : left : mid : right : tl)
565-
566-
-- | Convert a map to an unfoldable structure of key/value pairs where the keys are in ascending order
567-
toAscUnfoldable :: forall f k v. Unfoldable f => Map k v -> f (Tuple k v)
568-
toAscUnfoldable m = unfoldr go (m : Nil) where
569558
go Nil = Nothing
570559
go (hd : tl) = case hd of
571560
Leaf -> go tl
@@ -578,6 +567,23 @@ toAscUnfoldable m = unfoldr go (m : Nil) where
578567
Three left k1 v1 mid k2 v2 right ->
579568
go $ left : singleton k1 v1 : mid : singleton k2 v2 : right : tl
580569

570+
-- | Convert a map to an unfoldable structure of key/value pairs
571+
--
572+
-- While this traversal is up to 10% faster in benchmarks than `toUnfoldable`,
573+
-- it leaks the underlying map stucture, making it only suitable for applications
574+
-- where order is irrelevant.
575+
--
576+
-- If you are unsure, use `toUnfoldable`
577+
toUnfoldableUnordered :: forall f k v. Unfoldable f => Map k v -> f (Tuple k v)
578+
toUnfoldableUnordered m = unfoldr go (m : Nil) where
579+
go Nil = Nothing
580+
go (hd : tl) = case hd of
581+
Leaf -> go tl
582+
Two left k v right ->
583+
Just $ Tuple (Tuple k v) (left : right : tl)
584+
Three left k1 v1 mid k2 v2 right ->
585+
Just $ Tuple (Tuple k1 v1) (singleton k2 v2 : left : mid : right : tl)
586+
581587
-- | Get a list of the keys contained in a map
582588
keys :: forall k v. Map k v -> List k
583589
keys Leaf = Nil

test/Test/Data/Map.purs

+3-3
Original file line numberDiff line numberDiff line change
@@ -175,10 +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
181+
ascList = M.toUnfoldable m
182182
in ascList === sortBy (compare `on` fst) list
183183

184184
log "Lookup from union"
@@ -283,7 +283,7 @@ mapTests = do
283283

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

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

0 commit comments

Comments
 (0)