Introduction
In the 2018 Advent of Code challenged I solved all the puzzles in Groovy. It is pretty obvious, that choosing good data structure is the most important to obtain 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 testing performance 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
Loops types
Let’s implement the summing function in various ways.
collect
and sum
First 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 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 sum
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 from Java a big part of its syntax, 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
My tests I run 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, for small amount of iterations using built-in Groovy functions is good enough, but for much bigger amount of iterations we should use while
or for
loops like in plain, old Java.
Show me the code
Code for those examples are available here. You can run those examples on your machine and check performance on your own.