foreach(r; requests) { r.concider(); }

Steven Schveighoffer schveiguy at yahoo.com
Wed Oct 12 07:15:52 PDT 2011


On Wed, 12 Oct 2011 03:55:36 -0400, Gor Gyolchanyan  
<gor.f.gyolchanyan at gmail.com> wrote:

> The foreach loop is truly a marvelous tool, which allows one to
> implement custom iterations, which look and feel just like all other
> kinds of iterations.
> The only philosophical problem with it is, that it thinks that only
> classes and structs can be looped over in a custom way and that they
> can have only one way to be iterated over.
> It would be great to be able to implement iterative algorithms for
> arbitrary types (templated iterations), like strided iteration to
> extract the red part of an image.
> It would also be great to be able to have more, then one kind of
> iteration for every type of iterable.

You can already do this.

struct Iterable
{
    int opApply(int delegate(ref int) dg)
    {
       int result = 0;
       foreach(int i; 0..100)
       {
          auto t = i; // I hate this part of opApply BTW.
          if((result = dg(t)) != 0) break;
       }
       return result;
    }

    int inReverse(int delegate(ref int) dg)
    {
       int result = 0;
       foreach(int i; 0..100)
       {
          auto t = 99-i;
          if((result = dg(t)) != 0) break;
       }
       return result;
    }
}

void main()
{
    Interable it;
    foreach(i; it) {}
    foreach(i; &it.inReverse) {}
}

I've proposed an enhancement to make this better:

http://d.puremagic.com/issues/show_bug.cgi?id=2498

> Here's the list of what I mean by that:
> 1. Allow passing parameters to iterables in foreach:
>     foreach(c, i; MyType(), 3) { }
>     the `3` would be passed to MyType's opApply right after the
> delegate (of such an overload of opApply is available, of course).

This might be useful, but I don't like the syntax.

My preference would be to pass the parameters to the function itself, and  
infer the delegate from the foreach body.  i.e.:

struct Iterable
{
    int foo(int x, int delegate(ref int) dg) {...}
}

Iterable it;
foreach(i; it.foo(3)) {...}

Of course, this syntax is predicated on acceptance of my afore-mentioned  
enhancement request.

> 2. Allow named foreach loops:
>     foreach strided(c, i, MyType, 3) { }
>     the `strided` is passed as a template parameter to the opApplly,
> which (depending on what overloads of opApply are available) may be
> optional.

I don't see a large benefit of this over already existing foreach(c, i;  
&MyType.strided)

> 3. Allow free-function opApply, which could be templated to work with
> any kind of iterable in a specific way:
>     int opApply(string name : "strided", Iterable)(Iterable iterable,
> int delegate(ForeachParamsTuple!Iterable) dg, size_t stride) { /* ...
> */ }
>     this function would allow you to add stride to any iterable. the
> `ForeachParamsTuple` will return the tuple of parameters of the given
> iterable type.

Again, enhancement 2498 could be used for this.

> 4. Allow foreach loops with a single iterator to be specified without
> a body, in which case it would return an input range (or some other
> type of range), lazily evaluating and returning the iterator.
>     void printRange(Range)(Range range) { foreach(r, range) {   
> writeln(r); } };
>     unittest { printRange(foreach(i; 0..100)); }

This is not a good idea, since translating from an opApply loop to a range  
is not possible without spawning a new thread or copying the data.

The reason is simple -- foreach loops using opApply execute in the context  
of the opApply function, they cannot leave that context, and the context  
requires full use of the program stack.

I don't see a huge benefit of doing this vs.:

foreach(i; 0..100) writeln(i);

-Steve


More information about the Digitalmars-d mailing list