Welcome to ShenZhenJia Knowledge Sharing Community for programmer and developer-Open, Learning and Share
menu search
person
Welcome To Ask or Share your Answers For Others

Categories

Being somewhat new to the Java language I'm trying to familiarize myself with all the ways (or at least the non-pathological ones) that one might iterate through a list (or perhaps other collections) and the advantages or disadvantages of each.

(对Java语言有些陌生,我试图使自己熟悉所有可能遍历列表(或其他集合)的方式(或至少是非病理性方式)以及每种方式的优缺点。)

Given a List<E> list object, I know of the following ways to loop through all elements:

(给定一个List<E> list对象,我知道以下遍历所有元素的方式:)

Basic for loop (of course, there're equivalent while / do while loops as well) (基本的for 循环 (当然, while / do while循环也等效))

// Not recommended (see below)!
for (int i = 0; i < list.size(); i++) {
    E element = list.get(i);
    // 1 - can call methods of element
    // 2 - can use 'i' to make index-based calls to methods of list

    // ...
}

Note: As @amarseillan pointed out, this form is a poor choice for iterating over List s, because the actual implementation of the get method may not be as efficient as when using an Iterator .

(注意:正如@a??marseillan指出的那样,这种形式对于迭代List是一个糟糕的选择,因为get方法的实际实现可能不如使用Iterator时高效。)

For example, LinkedList implementations must traverse all of the elements preceding i to get the i-th element.

(例如, LinkedList实现必须遍历i之前的所有元素才能获得第i个元素。)

In the above example there's no way for the List implementation to "save its place" to make future iterations more efficient.

(在上面的示例中, List实现无法“保存位置”以使将来的迭代更加有效。)

For an ArrayList it doesn't really matter, because the complexity/cost of get is constant time (O(1)) whereas for a LinkedList is it proportional to the size of the list (O(n)).

(对于一个ArrayList它其实并不重要,因为复杂性/成本get是恒定的时间(O(1)),而对于LinkedList是成正比的列表的大小(为O(n))。)

For more information about the computational complexity of the built-in Collections implementations, check out this question .

(有关内置Collections实现的计算复杂度的更多信息,请查看此问题 。)

Enhanced for loop (nicely explained in this question ) (增强了for循环此问题对此做了很好的解释))

for (E element : list) {
    // 1 - can call methods of element

    // ...
}

Iterator (迭代器)

for (Iterator<E> iter = list.iterator(); iter.hasNext(); ) {
    E element = iter.next();
    // 1 - can call methods of element
    // 2 - can use iter.remove() to remove the current element from the list

    // ...
}

ListIterator (ListIterator)

for (ListIterator<E> iter = list.listIterator(); iter.hasNext(); ) {
    E element = iter.next();
    // 1 - can call methods of element
    // 2 - can use iter.remove() to remove the current element from the list
    // 3 - can use iter.add(...) to insert a new element into the list
    //     between element and iter->next()
    // 4 - can use iter.set(...) to replace the current element

    // ...
}

Functional Java (功能性Java)

list.stream().map(e -> e + 1); // Can apply a transformation function for e

Iterable.forEach , Stream.forEach , ... (Iterable.forEachStream.forEach ,...)

(A map method from Java 8's Stream API (see @i_am_zero's answer).)

((来自Java 8的Stream API的map方法(请参阅@i_am_zero的答案)。))

In Java 8 collection classes that implement Iterable (for example, all List s) now have a forEach method, which can be used instead of the for loop statement demonstrated above.

(在实现Iterable Java 8集合类(例如,所有List )中,现在具有forEach方法,可以使用该方法代替上面演示的for循环语句 。)

(Here is another question that provides a good comparison.)

((这是另一个可以很好比较的问题。))

Arrays.asList(1,2,3,4).forEach(System.out::println);
// 1 - can call methods of an element
// 2 - would need reference to containing object to remove an item
//     (TODO: someone please confirm / deny this)
// 3 - functionally separates iteration from the action
//     being performed with each item.

Arrays.asList(1,2,3,4).stream().forEach(System.out::println);
// Same capabilities as above plus potentially greater
// utilization of parallelism
// (caution: consequently, order of execution is not guaranteed,
// see [Stream.forEachOrdered][stream-foreach-ordered] for more
// information about this).

What other ways are there, if any?

(还有什么其他方式(如果有)?)

(BTW, my interest does not stem at all from a desire to optimize performance ; I just want to know what forms are available to me as a developer.)

((顺便说一句,我的兴趣根本不是出于优化性能的愿望;我只是想知道开发人员可以使用哪些形式。))

  ask by iX3 translate from so

与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
thumb_up_alt 0 like thumb_down_alt 0 dislike
299 views
Welcome To Ask or Share your Answers For Others

1 Answer

The three forms of looping are nearly identical.

(三种形式的循环几乎相同。)

The enhanced for loop:

(增强的for循环:)

for (E element : list) {
    . . .
}

is, according to the Java Language Specification , identical in effect to the explicit use of an iterator with a traditional for loop.

(根据Java语言规范 ,其作用与显式使用带有传统for循环的迭代器相同 。)

In the third case, you can only modify the list contents by removing the current element, and then only if you do it through the remove method of the iterator itself.

(在第三种情况下,您只能通过删除当前元素来修改列表内容,然后再通过迭代器本身的remove方法来进行操作。)

With index-based iteration, you are free to modify the list in any way.

(使用基于索引的迭代,您可以自由地以任何方式修改列表。)

However, adding or removing elements that come before the current index risks having your loop skipping elements or processing the same element multiple times;

(但是,添加或删除当前索引之前的元素可能会导致循环跳过元素或多次处理同一元素;)

you need to adjust the loop index properly when you make such changes.

(进行此类更改时,您需要适当地调整循环索引。)

In all cases, element is a reference to the actual list element.

(在所有情况下, element都是对实际list元素的引用。)

None of the iteration methods makes a copy of anything in the list.

(没有一种迭代方法可以复制列表中的任何内容。)

Changes to the internal state of element will always be seen in the internal state of the corresponding element on the list.

(element内部状态的更改将始终在列表中相应元素的内部状态中显示。)

Essentially, there only two ways to iterate over a list: by using an index or by using an iterator.

(本质上,只有两种方法可以遍历列表:使用索引或使用迭代器。)

The enhanced for loop is just a syntactic shortcut introduced in Java 5 to avoid the tedium of explicitly defining an iterator.

(增强的for循环只是Java 5中引入的语法快捷方式,以避免显式定义迭代器的繁琐工作。)

For both styles, you can come up with essentially trivial variations using for , while or do while blocks, but they all boil down to the same thing (or, rather, two things).

(对于这两种样式,您都可以使用forwhiledo while块提供一些琐碎的变体,但是它们全都归结为同一件事(或者说是两件事)。)

EDIT: As @iX3 points out in a comment, you can use a ListIterator to set the current element of a list as you are iterating.

(编辑:作为@ IX3在评论中指出,你可以使用ListIterator设置列表的当前元素为你迭代。)

You would need to use List#listIterator() instead of List#iterator() to initialize the loop variable (which, obviously, would have to be declared a ListIterator rather than an Iterator ).

(您将需要使用List#listIterator()而不是List#iterator()来初始化循环变量(显然,必须将其声明为ListIterator而不是Iterator )。)


与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
thumb_up_alt 0 like thumb_down_alt 0 dislike
Welcome to ShenZhenJia Knowledge Sharing Community for programmer and developer-Open, Learning and Share
...