在Java的for-each循环中是否有一种方法
for(String s : stringArray) { doSomethingWith(s); }
找出循环已经处理的频率?
除了使用旧的和众所周知的for(int i=0; i < boundary; i++)
循环之外,还有构造
int i = 0; for(String s : stringArray) { doSomethingWith(s); i++; }
在for-each循环中使用这种计数器的唯一方法是什么?
不,但你可以提供自己的柜台.
这样做的原因是,for-each循环在内部不具有一个计数器; 它基于Iterable接口,即它使用Iterator
循环遍历"集合" - 它可能根本不是集合,实际上可能根本不是基于索引(例如链表).
还有另一种方式.
鉴于您编写自己的Index
类和静态方法,可以返回Iterable
此类的实例
for (Indexeach: With.index(stringArray)) { each.value; each.index; ... }
执行的地方With.index
是什么样的
class With { public staticIterable > index(final T[] array) { return new Iterable >() { public Iterator > iterator() { return new Iterator >() { index = 0; public boolean hasNext() { return index < array.size } public Index next() { return new Index(array[index], index++); } ... } } } } }
最简单的解决方案是运行自己的计数器:
int i = 0; for (String s : stringArray) { doSomethingWith(s, i); i++; }
这样做的原因是因为没有实际保证集合中的项目(for
迭代的变体)甚至具有索引,甚至具有已定义的顺序(某些集合可能会在添加或删除元素时更改顺序).
例如,参见以下代码:
import java.util.*; public class TestApp { public static void AddAndDump(AbstractSetset, String str) { System.out.println("Adding [" + str + "]"); set.add(str); int i = 0; for(String s : set) { System.out.println(" " + i + ": " + s); i++; } } public static void main(String[] args) { AbstractSet coll = new HashSet (); AddAndDump(coll, "Hello"); AddAndDump(coll, "My"); AddAndDump(coll, "Name"); AddAndDump(coll, "Is"); AddAndDump(coll, "Pax"); } }
当你运行它时,你可以看到类似的东西:
Adding [Hello] 0: Hello Adding [My] 0: Hello 1: My Adding [Name] 0: Hello 1: My 2: Name Adding [Is] 0: Hello 1: Is 2: My 3: Name Adding [Pax] 0: Hello 1: Pax 2: Is 3: My 4: Name
表明,正确地说,顺序不被视为集合的显着特征.
还有其他方法可以在没有手动计数器的情况下完成,但这对于可疑的益处来说是一项相当大的工作.
在Java 8中使用lambdas和功能接口可以创建新的循环抽象.我可以使用索引和集合大小遍历集合:
Liststrings = Arrays.asList("one", "two","three","four"); forEach(strings, (x, i, n) -> System.out.println("" + (i+1) + "/"+n+": " + x));
哪个输出:
1/4: one 2/4: two 3/4: three 4/4: four
我实施的是:
@FunctionalInterface public interface LoopWithIndexAndSizeConsumer{ void accept(T t, int i, int n); } public static void forEach(Collection collection, LoopWithIndexAndSizeConsumer consumer) { int index = 0; for (T object : collection){ consumer.accept(object, index++, collection.size()); } }
可能性是无止境.例如,我创建了一个抽象,它仅对第一个元素使用特殊函数:
forEachHeadTail(strings, (head) -> System.out.print(head), (tail) -> System.out.print(","+tail));
其中正确打印逗号分隔列表:
one,two,three,four
我实施的是:
public staticvoid forEachHeadTail(Collection collection, Consumer headFunc, Consumer tailFunc) { int index = 0; for (T object : collection){ if (index++ == 0){ headFunc.accept(object); } else{ tailFunc.accept(object); } } }
图书馆将开始弹出来做这些事情,或者你可以自己动手.
Java 8引入了Iterable#forEach()
/ Map#forEach()
method,与"经典"for-each循环相比,它对许多Collection
/ Map
实现更有效.但是,在这种情况下也不提供索引.这里的技巧是AtomicInteger
在lambda表达式之外使用.注意:lambda表达式中使用的变量必须是有效的,这就是为什么我们不能使用普通的int
.
final AtomicInteger indexHolder = new AtomicInteger(); map.forEach((k, v) -> { final int index = indexHolder.getAndIncrement(); // use the index });
我担心这是不可能的foreach
.但我可以建议你一个简单的旧式for循环:
Listl = new ArrayList (); l.add("a"); l.add("b"); l.add("c"); l.add("d"); // the array String[] array = new String[l.size()]; for(ListIterator it =l.listIterator(); it.hasNext() ;) { array[it.nextIndex()] = it.next(); }
请注意,List接口允许您访问it.nextIndex()
.
(编辑)
对于你改变的例子:
for(ListIteratorit =l.listIterator(); it.hasNext() ;) { int i = it.nextIndex(); doSomethingWith(it.next(), i); }
其中一个变化Sun
是考虑Java7
提供对Iterator
foreach循环内部的访问.语法将是这样的(如果这是接受的):
for (String str : list : it) { if (str.length() > 100) { it.remove(); } }
这是语法糖,但显然对此功能提出了很多要求.但是在批准之前,你必须自己计算迭代次数,或者使用常规for循环Iterator
.
final Setdoubles; // boilerplate final Iterator iterator = doubles.iterator(); for (int ordinal = 0; iterator.hasNext(); ordinal++) { System.out.printf("%d:%f",ordinal,iterator.next()); System.out.println(); }
这实际上是谷歌在番石榴讨论中建议他们为什么不提供的解决方案
CountingIterator
.