Monday, November 19, 2007

Java Enhanced For Loops

Today, I am going to introduce one of the Java Technical Tips that I have previously received from sun on my e-mail. I wanted to share it with you . I hope every one who reads this tip will get benefit from.

The author of this tip is John Zukowski, president and principal consultant of JZ Ventures, Inc.
This tip was developed using Java SE 6.


Using Enhanced For-Loops with Your Classes

The enhanced for-loop is a popular feature introduced with the Java SE platform in version 5.0. Its simple structure allows one to simplify code by presenting for-loops that visit each element of an array/collection without explicitly expressing how one goes from element to element.

Because the old style of coding didn't become invalid with the new for-loop syntax, you don't have to use an enhanced for-loop when visiting each element of an array/collection. However, with the new style, one's code would typically change from something like the following:

for (int i=0; i
System.out.println("Element: " + array[i]);
}

to the newer form:

for (String element : array) {
System.out.println("Element: " + element);
}

Assuming "array" is defined to be an array of String objects, each element is assigned to the element variable as it loops through the array. These basics of the enhanced for-loop were covered in an earlier Tech Tip: The Enhanced For Loop, from May 5, 2005.

If you have a class called Colony which contains a group of Penguin objects, without doing anything extra to get the enhanced for-loop to work, one way you would loop through each penguin element would be to return an Iterator and iterate through the colony. Unfortunately, the enhanced for-loop does not work with Iterator , so the following won't even compile:


// Does not compile
import java.util.*;
public class BadColony {
static class Penguin {
String name;
Penguin(String name) {
this.name = name;
}
public String toString() {
return "Penguin{" + name + "}";
}
}

Set set = new HashSet();

public void addPenguin(Penguin p) {
set.add(p);
}

public Iterator getPenguins() {
return set.iterator();
}

public static void main(String args[]) {
Colony colony = new Colony();
Penguin opus = new Penguin("Opus");
Penguin chilly = new Penguin("Chilly Willy");
Penguin mumble = new Penguin("Mumble");
Penguin emperor = new Penguin("Emperor");
colony.addPenguin(opus);
colony.addPenguin(chilly);
colony.addPenguin(mumble);
colony.addPenguin(emperor);
Iterator it = colony.getPenguins();
// The bad line of code:
for (Penguin p : it) {
System.out.println(p);
}
}
}

You cannot just pass an Iterator into the enhanced for-loop. The 2nd line of the following will generate a compilation error:

Iterator it = colony.getPenguins();
for (Penguin p : it) {

The error:

BadColony.java:36: foreach not applicable to expression type
for (Penguin p : it) {
^
1 error

In order to be able to use your class with an enhanced for-loop, it does need an Iterator , but that Iterator must be provided via the Iterable interface:


public interface java.lang.Iterable {
public java.util.Iterator iterator();
}

Actually, to be more correct, you can use a generic T , allowing the enhanced for-loop to avoid casting, returning the designated generic type, instead of just a plain old Object .

public interface java.lang.Iterable {
public java.util.Iterator iterator();
}

It is this Iterable object which is then provided to the enhanced for-loop. By making the Colony class implement Iterable , and having its new iterator()Iterator that getPenguins() provides, you'll be able to loop through the penguins in the colony via an enhanced for-loop. method return the

By adding the proper implements clause:

public class Colony implements Iterable {


You then get your enhanced for
-loop for the colony:

for (Penguin p : colony) {

Here's the updated Colony class with the corrected code:

import java.util.*;

public class Colony implements Iterable {

static class Penguin {
String name;
Penguin(String name) {
this.name = name;
}
public String toString() {
return "Penguin{" + name + "}";
}
}

Set set = new HashSet();

public void addPenguin(Penguin p) {
set.add(p);
}

public Iterator getPenguins() {
return set.iterator();
}

public Iterator iterator() {
return getPenguins();
}

public static void main(String args[]) {
Colony colony = new Colony();
Penguin opus = new Penguin("Opus");
Penguin chilly = new Penguin("Chilly Willy");
Penguin mumble = new Penguin("Mumble");
Penguin emperor = new Penguin("Emperor");
colony.addPenguin(opus);
colony.addPenguin(chilly);
colony.addPenguin(mumble);
colony.addPenguin(emperor);
for (Penguin p : colony) {
System.out.println(p);
}
}
}

Running the code produces the
following output:

> java Colony

Penguin{Chilly Willy}
Penguin{Mumble}
Penguin{Opus}
Penguin{Emperor}

Keep in mind that the individual penguins are internally kept in

a Set type collection so the returned order doesn't necessarily match

the insertion order, which in this case it doesn't.

Remember to genericize the implements clause for the class

"implements Iterable " and not just say "implements Iterable ".

With the latter, the enhanced for-loop will only return an Object for

each element.

For more information on the enhanced for-loop, please see the

Java Programming Language guide from JDK 1.5.

No comments: