Why can't I use foreach on Java Enumeration?
Why can't I do:
Enumeration e = ...
for (Object o : e)
...
Because Enumeration<T>
doesn't extend Iterable<T>
. Here is an example of making Iterable Enumerations.
As to why that's an interesting question. This isn't exactly your question but it sheds some light on it. From the Java Collections API Design FAQ:
Why doesn't Iterator extend Enumeration?
We view the method names for Enumeration as unfortunate. They're very long, and very frequently used. Given that we were adding a method and creating a whole new framework, we felt that it would be foolish not to take advantage of the opportunity to improve the names. Of course we could support the new and old names in Iterator, but it doesn't seem worthwhile.
That basically suggests to me that Sun wants to distance themselves from Enumeration, which is very early Java with quite a verbose syntax.
Using Collections utility class, Enumeration can be made iterable like:
Enumeration headerValues=request.getHeaders("mycustomheader");
List headerValuesList=Collections.list(headerValues);
for(Object headerValueObj:headerValuesList){
... do whatever you want to do with headerValueObj
}
I have solved this problem with two very simple classes, one for Enumeration
and one for Iterator
. The enumeration wrapper is as follows:
static class IterableEnumeration<T>
extends Object
implements Iterable<T>, Iterator<T>
{
private final Enumeration<T> enumeration;
private boolean used=false;
IterableEnumeration(final Enumeration<T> enm) {
enumeration=enm;
}
public Iterator<T> iterator() {
if(used) { throw new IllegalStateException("Cannot use iterator from asIterable wrapper more than once"); }
used=true;
return this;
}
public boolean hasNext() { return enumeration.hasMoreElements(); }
public T next() { return enumeration.nextElement(); }
public void remove() { throw new UnsupportedOperationException("Cannot remove elements from AsIterator wrapper around Enumeration"); }
}
Which can be used either with a static utility method (which is my preference):
/**
* Convert an `Enumeration<T>` to an `Iterable<T>` for a once-off use in an enhanced for loop.
*/
static public <T> Iterable<T> asIterable(final Enumeration<T> enm) {
return new IterableEnumeration<T>(enm);
}
...
for(String val: Util.asIterable(enm)) {
...
}
or by instantiating the class:
for(String val: new IterableEnumeration<String>(enm)) {
...
}
The new-style-for-loop ("foreach") works on arrays, and things that implement the Iterable
interface.
It's also more analogous to Iterator
than to Iterable
, so it wouldn't make sense for Enumeration
to work with foreach unless Iterator
did too (and it doesn't).
Enumeration
is also discouraged in favor of Iterator
.
With java 8 and beyond this is possible:
import java.util.Collections;
import java.util.Enumeration;
Enumeration e = ...;
Collections.list(e).forEach(o -> {
... // use item "o"
});