Idiomatic way to use for-each loop given an iterator?

后端 未结 9 1589
星月不相逢
星月不相逢 2020-12-03 09:52

When the enhanced for loop (foreach loop) was added to Java, it was made to work with a target of either an array or Iterable.

for ( T item : /*         


        
相关标签:
9条回答
  • 2020-12-03 10:33
    public class DescendingIterableDequeAdapter<T> implements Iterable<T> {
        private Deque<T> original;
    
        public DescendingIterableDequeAdapter(Deque<T> original) {
            this.original = original;
        }
    
        public Iterator<T> iterator() {
             return original.descendingIterator();
        }
    }
    

    And then

    for (T item : new DescendingIterableDequeAdapter(deque)) {
    
    }
    

    So, for each such case, you'd need a special adapter. I don't think it is theoretically possible to do what you want, because the facility has to know what iterator-returning methods exist, so that it can call them.

    As for your additional question - I believe because the for-each loop was actually meant to make things shorter for general-purpose scenarios. And calling an additional method makes the syntax more verbose. It could've supported both Iterable and Iterator, but what if the object passed implemented both? (would be odd, but still possible).

    0 讨论(0)
  • 2020-12-03 10:33

    I suggest to create a helper class with factory methods that you can use like this:

    import static Iter.*;
    
    for( Element i : iter(elements) ) {
    }
    
    for( Element i : iter(o, Element.class) ) {
    }
    

    As a next step, the return type of iter() could be a fluent interface so you can do:

    for( Element i : iter(elements).reverse() ) {
    }
    

    or maybe

    for( Element i : reverse(elements) ) {
    }
    

    You should also have a look at op4j which solves many of these problems with a very nice API.

    0 讨论(0)
  • 2020-12-03 10:38

    Guava users can do ImmutableList.copyOf(Iterator) to safely convert an Iterator into an Iterable. Despite the seeming simplicity of looping over an Iterator, there are concerns that foreach hides, and the safest option is to create a stable data structure like a list.

    This is also discussed in the Idea Graveyard:

    The biggest concern is that Iterable is generally assumed to be able to produce multiple independent iterators. The doc doesn't say this, but the Collection doc doesn't say this, either, and yet we assume it of its iterators. We have had breakages in Google when this assumption was violated.

    The simplest workaround is ImmutableList.copyOf(Iterator), which is pretty fast, safe, and provides many other advantages besides.

    0 讨论(0)
提交回复
热议问题