Advent of Code 2024 day 2 in Zig
Published: 01-14-25
Advent of Code
- in my previous blog post I explained advent of code. In essence it’s a great way to learn a new language and practice file / string manipulations.
- the problem for today can be found here.
Zig Implementation
- for today’s problem we can’t do a simple implementation because part 2 of the problem involves checking the ‘safety’ of a line if one number is removed from it.
- this means we have to check each ‘unsafe’ line n times, and because we have to go through each line in the input our total runtime is
O(n^2)
. - to check each line we have to reach for the
ArrayList
data structure, but we can save time and memory by using the.items
field to view a slice of the elements.
- this means we have to check each ‘unsafe’ line n times, and because we have to go through each line in the input our total runtime is
Parsing the file
- today I opted for the more simple
@embed
to get the file converted into a[]u8
buffer compiled into the binary.
// embed the file
const input = @embedFile("input.txt");
// convert it into an iterator
var it = mem.tokenizeScalar(u8, input, '\n');
Function for defining whether a line is ‘safe’
- if you haven’t followed the link to the problem it is essentially a problem of comparing numbers on the same line.
- our input is a file with multiple lines, each of which is composed of several numbers
- a line is ’safe’ all of the numbers either all increase by at most 3, or decrease by at most 3.
- for part2 we have to check if removing a number from a line makes it ’safe’
- instead of operating on an ArrayList we will operate on slices this is a little more performant and we can ignore a specific index rather easily.
// helper function to get absolute value of an integer
pub fn abs(x: i32) i32 {
if (x < 0) {
return x * -1;
}
return x;
}
// determine if a line is 'safe'
// takes a line of numbers and an index to ignore
pub fn isSafe(line: []i32, i: usize) bool {
var lastNum: i32 = -1;
var lastDif: i32 = 0;
for (line, 0..) |num, index| {
// first number in line
if (index == i) continue;
if (lastNum < 0) {
lastNum = num;
} else {
// current difference
const diff = (num - lastNum);
// check for 'safety'
if (diff == 0 or // no change
abs(diff) > 3 or // inc or dec greater than 3
(lastDif < 0 and diff > 0) or // switch from dec to inc
(lastDif > 0 and diff < 0)) // switch from inc to dec
{
return false; // not a safe line
}
// set last difference and number
lastDif = (num - lastNum);
lastNum = num;
}
}
return true;
}
putting all the pieces together in our ‘main’ function
pub fn main() !void {
const allocator = std.heap.page_allocator;
std.debug.print("AOC 2024 day 2\n", .{});
var it = mem.tokenizeScalar(u8, input, '\n');
// scores for part 1 and 2
var p1: i32 = 0;
var p2: i32 = 0;
while (it.next()) |line| {
// get individual numbers
var nums = mem.tokenizeScalar(u8, line, ' ');
// ArrayList to hold number values in the line
var lineNums = ArrayList(i32).init(allocator);
defer lineNums.deinit();
// Populate the ArrayList O(n)
while (nums.next()) |numu8| {
const num = try std.fmt.parseInt(i32, numu8, 10);
try lineNums.append(num);
}
// if safe add line for p1
if (isSafe(lineNums.items, lineNums.items.len + 1)) { // O(n)
p1 += 1;
p2 += 1;
} else {
// if safe with one level removed add line for p2
for (0..lineNums.items.len) |i| { // O(n)
// take the 'items' field from ArrayList and give the index to ignore
if (isSafe(lineNums.items, i)) { // O(n)
p2 += 1;
break;
}
} // total runtime = O(n * n) + O(n) + O(n) = O(n^2)
}
}
// print answer for part1
std.debug.print("Part1 => {d}\n", .{p1});
// print answer for part2
std.debug.print("Part2 => {d}\n", .{p2});
}