# NAME

List::Lazy - Generate lists lazily

# VERSION

version 0.3.2

# SYNOPSIS

```perl
use List::Lazy qw/ lazy_range /;

my $range = lazy_range( 1, undef )->grep(sub{ $_ % 2})->map( sub { '!' x $_ } );

say $_ for $range->next(3); # prints ! !!! !!!!!
```

# DESCRIPTION

`List::Lazy` creates lists that lazily evaluate their next values on-demand.

# EXPORTED FUNCTIONS

Lazy::List doesn't export any function by default, but will export the three following 
functions on request.

## lazy\_list

```perl
my $list  = lazy_list $generator_sub, $state;
```

A convenience shortcut for the List::Lazy constructor. The `$state` will be available
(and can be changed) by the generator subroutine. The generator subroutine is expected
to return a list of one or more next items of the list. Returning an empty list means
that the list has reached its end.

```perl
my $even_numbers = lazy_list { $_ += 2 } 0; # will return 2, 4, 6, ...
```

In additional of regular values, the generator can also return lazy lists,
which will be seamlessly expanded.

```perl
my $list = lazy_range( 1, undef )->map(sub { lazy_range( 1, $_ ) });
# will return 1, 1, 2, 1, 2, 3, 1, 2, 3, 4, ...
```

## lazy\_range

```perl
my $range = lazy_range $min, $max, $iterator;
```

Creates a list iterating over a range of values. `$min` and `$max` are required, but `$max`  can be 
`undef` (meaning no upper limit). The `$iterator` is optional and defaults to the value `1`. 
The `$iterator` can be a number, which will be the step at which the numbers are increased, or a coderef that will 
be passed the previous value as `$_`, and is expected to return the next value.

```perl
my $palinumbers = lazy_range 99, undef, sub { do { $_++ } until $_ eq reverse $_; $_ };

say join ' ', $palinumbers->next(3); # 99 101 111
```

## lazy\_fixed\_list

```perl
my $list = lazy_fixed_list @some_array;
```

Creates a lazy list that will returns the values of the given array. 

# CLASS

## new

```perl
my $list = List::Lazy->new(
    state => 1,
    generator => sub {
        $_++;
    },
);
```

Creates a lazy list.

### arguments

- state

    The state will be passed to the generator as `$_`. If it is modified by the generator,
    its new value will be saved for the next invocation.

- generator

    A coderef that generates one or more next items for the list. If it returns an empty list,
    the stream will be considered to be exhausted.

## is\_done

Returns `true` is the list is exhausted.

## next($num)

Returns the next `$num` items of the list (or less if the list doesn't have
that many items left). `$num` defaults to `1`.

```perl
my $range = lazy_range 1, 100;

while( my $next = $range->next ) {
    ...
}
```

## reduce

```perl
my $value = $list->reduce( $reducing_sub, $initial_value );
```

Iterates through the list and reduces its values via the `$reducing_sub`, which
will be passed the cumulative value and the next item via `$a` and `$b`. 
If `$initial_value` is not given, it defaults to the first element of the list.

```perl
my $sum = lazy_range( 1, 100 )->reduce( sub { $a + $b } );
```

## batch

```perl
my $new_list = $list->batch($n);
```

Creates a new list where the items of the original list are batched in groups
of `$n` (or less for the last batch).

```perl
my $list = lazy_fixed_list( 1..100 )->batch(3);

my $x = $list->next;           # $x == [ 1, 2, 3]
```

## map

```perl
my $new_list = $list->map( $mapper_sub );
```

Creates a new list by applying the transformation given by `$mapper_sub` to the
original list. The sub ill be passed the original next item via `$_`
and is expected to return its transformation, which 
can modify the item, explode it into many items, or suppress it,

Note that the new list do a deep clone of the original list's state, so reading
from the new list won't affect the original list.

```perl
my $recount = ( lazy_range 1, 100 )->map( sub { 1..$_ } );
# will return 1 1 2 1 2 3 1 2 3 4 ...
```

## grep

```perl
my $new_list = $list->grep( $filter_sub );
```

Creates a new list by applying the filtering given by `$filter_sub` to the
original list. The sub will be passed the original next item via `$_`
and is expected to return a boolean indicating if the item should be kept or not.

Note that the new list do a deep clone of the original list's state, so reading
from the new list won't affect the original list.

```perl
my $odd = ( lazy_range 1, 100 )->grep( sub { $_ % 2 } );
```

## spy

```perl
my $new_list = $list->spy( $sub );
```

Creates a new list that will execute the spy `$sub` for
every value it sees (with the value assigned to `$_`). 

If `$sub` is not given, it'll `carp` the values.

## until

```perl
my $new_list = $list->until( $condition );
```

Creates a new list that truncates the original list as soon
as the condition is met.

```perl
my $to_ten = $list->until(sub{ $_ > 10 });
```

## append

```perl
my $new_list = $list->append( @other_lists );
```

Creates a new list that will return first the elements of `$list`,
and those of the `@other_lists`. 

Note that the new list do a deep clone of the original lists's state, so reading
from the new list won't affect the original lists.

```perl
my $range = lazy_range 1..100;
my $twice = $range->append( $range );
```

## prepend

```perl
my $new_list = $list->prepend( @other_lists );
```

Like `append`, but prepend the other lists to the current one.

Note that the new list do a deep clone of the original lists's state, so reading
from the new list won't affect the original lists.

## all

```perl
my @rest = $list->all;
```

Returns all the remaining values of the list. Be careful: if the list is unbounded, 
calling `all()` on it will result into an infinite loop.

# AUTHOR

Yanick Champoux <yanick@babyl.dyndns.org> [![endorse](http://api.coderwall.com/yanick/endorsecount.png)](http://coderwall.com/yanick)

# COPYRIGHT AND LICENSE

This software is copyright (c) 2019, 2018, 2017, 2016 by Yanick Champoux.

This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.