Monadic flatMap combinator #
This file provides the flatMap iterator combinator and variants of it.
If it is any iterator, it.flatMap f maps each output of it to an iterator using
f.
it.flatMap f first emits all outputs of the first obtained iterator, then of the second,
and so on. In other words, it flattens the iterator of iterators obtained by mapping with
f.
Let it₁ and it₂ be iterators and f a monadic function mapping it₁'s outputs to iterators
of the same type as it₂. Then it₁.flatMapAfterM f it₂ first goes over it₂ and then over
it₁.flatMap f it₂, emitting all their values.
The main purpose of this combinator is to represent the intermediate state of a flatMap iterator
that is currently iterating over one of the inner iterators.
Marble diagram (without monadic effects):
it₁                            --b      c    --d -⊥
it₂                      a1-a2⊥
f b                               b1-b2⊥
f c                                      c1-c2⊥
f d                                             ⊥
it.flatMapAfterM f it₂   a1-a2----b1-b2--c1-c2----⊥
Termination properties:
- Finiteinstance: only if- it₁,- it₂and the inner iterators are finite
- Productiveinstance: only if- it₁is finite and- it₂and the inner iterators are productive
For certain functions f, the resulting iterator will be finite (or productive) even though
no Finite (or Productive) instance is provided out of the box. For example, if the outer
iterator is productive and the inner
iterators are productive and provably never empty, then the resulting iterator is also productive.
Performance:
This combinator incurs an additional O(1) cost with each output of it₁, it₂ or an internal
iterator.
For each value emitted by the outer iterator it₁, this combinator calls f.
Equations
- Std.Iterators.IterM.flatMapAfterM f it₁ it₂ = Std.Iterators.IterM.flattenAfter✝ (Std.Iterators.IterM.mapM f it₁) it₂
Instances For
Let it be an iterator and f a monadic function mapping it's outputs to iterators.
Then it.flatMapM f is an iterator that goes over it and for each output, it applies f and
iterates over the resulting iterator. it.flatMapM f emits all values obtained from the inner
iterators -- first, all of the first inner iterator, then all of the second one, and so on.
Marble diagram (without monadic effects):
it                 ---a      --b      c    --d -⊥
f a                    a1-a2⊥
f b                             b1-b2⊥
f c                                    c1-c2⊥
f d                                           ⊥
it.flatMapM        ----a1-a2----b1-b2--c1-c2----⊥
Termination properties:
- Finiteinstance: only if- itand the inner iterators are finite
- Productiveinstance: only if- itis finite and the inner iterators are productive
For certain functions f, the resulting iterator will be finite (or productive) even though
no Finite (or Productive) instance is provided out of the box. For example, if the outer
iterator is productive and the inner
iterators are productive and provably never empty, then the resulting iterator is also productive.
Performance:
This combinator incurs an additional O(1) cost with each output of it or an internal iterator.
For each value emitted by the outer iterator it, this combinator calls f.
Equations
Instances For
Let it₁ and it₂ be iterators and f a function mapping it₁'s outputs to iterators
of the same type as it₂. Then it₁.flatMapAfter f it₂ first goes over it₂ and then over
it₁.flatMap f it₂, emitting all their values.
The main purpose of this combinator is to represent the intermediate state of a flatMap iterator
that is currently iterating over one of the inner iterators.
Marble diagram:
it₁                            --b      c    --d -⊥
it₂                      a1-a2⊥
f b                               b1-b2⊥
f c                                      c1-c2⊥
f d                                             ⊥
it.flatMapAfter  f it₂   a1-a2----b1-b2--c1-c2----⊥
Termination properties:
- Finiteinstance: only if- it₁,- it₂and the inner iterators are finite
- Productiveinstance: only if- it₁is finite and- it₂and the inner iterators are productive
For certain functions f, the resulting iterator will be finite (or productive) even though
no Finite (or Productive) instance is provided out of the box. For example, if the outer
iterator is productive and the inner
iterators are productive and provably never empty, then the resulting iterator is also productive.
Performance:
This combinator incurs an additional O(1) cost with each output of it₁, it₂ or an internal
iterator.
For each value emitted by the outer iterator it₁, this combinator calls f.
Equations
- Std.Iterators.IterM.flatMapAfter f it₁ it₂ = Std.Iterators.IterM.flattenAfter✝ (Std.Iterators.IterM.map f it₁) it₂
Instances For
Let it be an iterator and f a function mapping it's outputs to iterators.
Then it.flatMap f is an iterator that goes over it and for each output, it applies f and
iterates over the resulting iterator. it.flatMap f emits all values obtained from the inner
iterators -- first, all of the first inner iterator, then all of the second one, and so on.
Marble diagram:
it                 ---a      --b      c    --d -⊥
f a                    a1-a2⊥
f b                             b1-b2⊥
f c                                    c1-c2⊥
f d                                           ⊥
it.flatMap         ----a1-a2----b1-b2--c1-c2----⊥
Termination properties:
- Finiteinstance: only if- itand the inner iterators are finite
- Productiveinstance: only if- itis finite and the inner iterators are productive
For certain functions f, the resulting iterator will be finite (or productive) even though
no Finite (or Productive) instance is provided out of the box. For example, if the outer
iterator is productive and the inner
iterators are productive and provably never empty, then the resulting iterator is also productive.
Performance:
This combinator incurs an additional O(1) cost with each output of it or an internal iterator.
For each value emitted by the outer iterator it, this combinator calls f.
Equations
Instances For
The plausible-step predicate for Flatten iterators
- outerYield {α α₂ β : Type w} {m : Type w → Type w'} [Iterator α m (IterM m β)] [Iterator α₂ m β] {it₁ it₁' : IterM m (IterM m β)} {it₂' : IterM m β} : it₁.IsPlausibleStep (IterStep.yield it₁' it₂') → IsPlausibleStep (toIterM { it₁ := it₁, it₂ := none } m β) (IterStep.skip (toIterM { it₁ := it₁', it₂ := some it₂' } m β))
- outerSkip {α α₂ β : Type w} {m : Type w → Type w'} [Iterator α m (IterM m β)] [Iterator α₂ m β] {it₁ it₁' : IterM m (IterM m β)} : it₁.IsPlausibleStep (IterStep.skip it₁') → IsPlausibleStep (toIterM { it₁ := it₁, it₂ := none } m β) (IterStep.skip (toIterM { it₁ := it₁', it₂ := none } m β))
- outerDone {α α₂ β : Type w} {m : Type w → Type w'} [Iterator α m (IterM m β)] [Iterator α₂ m β] {it₁ : IterM m (IterM m β)} : it₁.IsPlausibleStep IterStep.done → IsPlausibleStep (toIterM { it₁ := it₁, it₂ := none } m β) IterStep.done
- innerYield {α α₂ β : Type w} {m : Type w → Type w'} [Iterator α m (IterM m β)] [Iterator α₂ m β] {it₁ : IterM m (IterM m β)} {it₂ it₂' : IterM m β} {b : β} : it₂.IsPlausibleStep (IterStep.yield it₂' b) → IsPlausibleStep (toIterM { it₁ := it₁, it₂ := some it₂ } m β) (IterStep.yield (toIterM { it₁ := it₁, it₂ := some it₂' } m β) b)
- innerSkip {α α₂ β : Type w} {m : Type w → Type w'} [Iterator α m (IterM m β)] [Iterator α₂ m β] {it₁ : IterM m (IterM m β)} {it₂ it₂' : IterM m β} : it₂.IsPlausibleStep (IterStep.skip it₂') → IsPlausibleStep (toIterM { it₁ := it₁, it₂ := some it₂ } m β) (IterStep.skip (toIterM { it₁ := it₁, it₂ := some it₂' } m β))
- innerDone {α α₂ β : Type w} {m : Type w → Type w'} [Iterator α m (IterM m β)] [Iterator α₂ m β] {it₁ : IterM m (IterM m β)} {it₂ : IterM m β} : it₂.IsPlausibleStep IterStep.done → IsPlausibleStep (toIterM { it₁ := it₁, it₂ := some it₂ } m β) (IterStep.skip (toIterM { it₁ := it₁, it₂ := none } m β))