for-loops: structured serial iteration

For-loops are Chapel’s means of expressing serial iteration as driven by an iterand—an iterator function or iterable expression. As a simple example, the following code loops over the integers 1 through 5, printing a message per iteration:

examples/users-guide/base/forloops.chpl
for i in 1..5 do
  writeln("hello #", i);

More specifically, this for-loop invokes the default iterator for the range expression 1..5, storing the values that it yields in a per-iteration variable named i. The body of the loop then refers to i as a means of customizing the messages that are printed. The resulting output is:

hello #1
hello #2
hello #3
hello #4
hello #5

For-loop Index Variables

A for-loop’s index variable (like i above) is a new symbol that the for-loop defines, permitting its body to refer to the values generated by its iterand expression. Thus, the following code actually declares two variables named i where the for-loop’s index variable temporarily shadows the original i and has no relation to it:

var i: int;
for i in 1..5 do
  writeln(i);
writeln("i is: ", i);

The original variable i remains unchanged by the loop, so it retains its default value once it is no longer shadowed. Thus, the output for the program is:

1
2
3
4
5
i is: 0

A for-loop’s index variable may or may not be mutable, depending on the expression that drives the loop. For example, range iterators yield const integer values, making it illegal to modify i within the loop body:

for i in 1..5 do
  i += 1;        // Illegal: i is 'const' for ranges

In contrast, an array’s default iterator yields references to the array’s elements. As a result, the loop’s index variable can be used as a means of modifying the array’s values:

var A: [1..5] real;
for a in A do
  a = 1.2;

Thus, printing out the array after this loop terminates would reflect the updated values:

1.2 1.2 1.2 1.2 1.2

If the values yielded by the iterand are not required by the loop body, the index variable and in keyword can be omitted altogether. For example, the following loop iterates five times, but since the index variable is not required by the body, it has been omitted:

writeln("Printing five messages:");
for 1..5 do
  writeln("hi!");

For-loop Bodies

If the body of a for-loop is a single statement, as in all of the examples above, the do keyword can be used after the iterand to specify the loop body. Multi-statement loop bodies can be defined using curly brackets:

for i in 1..5 {
  writeln("Computing on A[", i, "]");
  A[i] += i;
}

Of course, since a compound statement is a single statement itself, this also means that the two styles can be combined (though with no real benefit):

for i in 1..5 do {
  writeln("Computing on A[", i, "]");
  A[i] += (i / 10.0);
}

Additional For-loop Capabilities

For loops can be used to iterate over multiple expressions in a coordinated or zippered manner, described on the next page. Subsequent pages will also describe how to write iterator functions. Note that it is also possible to exit early from a for-loop via the break statement, covered later.