Loop Performance in Groovy
Learn more about measuring loop performance in Groovy.
Join the DZone community and get the full member experience.
Join For FreeIn the 2018 Advent of Code challenged, I solved all the puzzles in Groovy. It is pretty obvious that choosing a good data structure is the most important part of obtaining a performant solution. However, the way we iterate over those structures is also very significant, at least when using Groovy.
Measuring Performance
I want to measure how long it takes to sum some numbers. For the performance testing of loops, I prepared a small function that simply sums some numbers:
void printAddingTime(String message, long to, Closure<Long> adder) {
LocalTime start = LocalTime.now()
long sum = adder(to)
println("$message: $sum calculated in ${Duration.between(start, LocalTime.now()).toMillis()} ms")
}
Pseudo code for summing functions is below:
for i = 1 to n
for j = 1 to n
sum += i * j
end
end
Loop Types
Let's implement the summing function in various ways.
collect
and sum
First, the loop type is to use built-in (by Groovy) function collect
and sum
on collections (Range
it this example):
(1..n).collect { long i ->
(1..n).collect { long j ->
i * j
}.sum()
}.sum()
each
Next, let's write the same function using the each
built-in function on collections (Range
it this example) and then add results to accumulator variable:
long sum = 0
(1..n).each { long i ->
(1..n).each { long j ->
sum += i * j
}
}
return sum
times
Now, instead of using each
, we could use the function times
built-in on Number
by Groovy:
long sum = 0
n.times { long i ->
n.times { long j ->
sum += (i + 1)*(j+1)
}
}
return sum
We have to add 1
to i
and j
because times generates numbers from 0
to n
exclusive.
LongStream
With sum
Java 8 came with a new feature — streams. One example of streams is LongStream
. Fortunately, it has asum
built-in function, which we can use:
LongStream.range(0, n).map { i ->
LongStream.range(0, n).map { j ->
(i + 1) * (j + 1)
}.sum()
}.sum()
LongStream
generates numbers in the same way as times
function, so we also have to add 1
to i
and j
here.
LongStream
With Manual Sum
Instead of sum
function on LongStream
, we can add all numbers manually:
long sum = 0
LongStream.range(0, n).forEach { i ->
LongStream.range(0, n).forEach { j ->
sum += (i + 1) * (j + 1)
}
}
return sum
while
Of course, since Groovy inherits a big part of its syntax from Java, we can use the while
loop:
long sum = 0
long i = 1
while(i <= n){
long j = 1
while(j <= n){
sum+= i*j
++j
}
++i
}
return sum
for
As we can use while
, we can also use for
loop in Groovy:
long sum = 0
for (long i = 1; i <= n; ++i) {
for (long j = 1; j <= n; ++j) {
sum += i * j
}
}
return sum
Results
For my tests, I ran on Java 1.8
and Groovy 2.5.5
. Script loops.groovy
was fired using bash script:
#!/bin/sh
for x in 10 100 1000 10000 100000; do
echo $x
groovy loops.groovy $x
echo
done
Values are in milliseconds:
Loop n | 10 | 100 | 1000 | 10000 | 100000 |
---|---|---|---|---|---|
collect + sum |
7 | 22 | 216 | 16244 | 1546822 |
each |
12 | 17 | 118 | 7332 | 706781 |
times |
2 | 10 | 109 | 8264 | 708684 |
LongStream + sum |
7 | 17 | 127 | 7679 | 763341 |
LongStream + manual sum |
18 | 35 | 149 | 6857 | 680804 |
while |
8 | 20 | 103 | 3166 | 301967 |
for |
7 | 10 | 25 | 359 | 27966 |
As you can spot, although the small number of iterations using built-in Groovy functions is good enough, for a much bigger amount of iterations, we should use while
or for
loops as in plain, old Java.
Code for these examples is available here. You can run those examples on your machine and check performance on your own.
Happy looping!
Published at DZone with permission of Dominik Przybysz. See the original article here.
Opinions expressed by DZone contributors are their own.
Comments