Why can't I do:
Enumeration e = ...
for (Object o : e)
...
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 using the adapter pattern.
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 the 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
}
This uses the Collections.list
method, which copies the Enumeration
data into a new ArrayList
.
Returns an array list containing the elements returned by the specified enumeration in the order they are returned by the enumeration. This method provides interoperability between legacy APIs that return enumerations and new APIs that require collections.
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"
});
Enumeration
doesn't implement Iterable
and as such can't be used directly in a foreach loop. However using Apache Commons Collections it's possible to iterate over an enumeration with:
for (Object o : new IteratorIterable(new EnumerationIterator(e))) {
...
}
You could also use a shorter syntax with Collections.list()
but this is less efficient (two iterations over the elements and more memory usage) :
for (Object o : Collections.list(e))) {
...
}
The Enumeration
legacy class is analogous to the modern Iterator
class. Both represent a current iteration over data.
The enhanced for loop was designed to work with Iterable
and arrays, but notably not with Iterator
. Iterable
provides an iterator
method, which returns an Iterator
over its data.
While it isn't possible to use the enhanced for loop with Enumeration
or its modern Iterator
equivalent, the forEachRemaining
method in Iterator
(added in Java 8) provides similar functionality. Combining with Enumeration.asIterator
(added in Java 9) allows iterating over an enumeration in a similar manner as the enhanced for loop:
Enumeration<Object> e = ...
e.asIterator().forEachRemaining(o -> {
// ...
});
Because an Enumeration (and most classes derived from this interface) does not implement Iterable.
You can try to write your own wrapper class.