AdventOfCode/2024/12/solution.py

204 lines
6.7 KiB
Python
Raw Permalink Normal View History

2024-12-12 14:20:12 +01:00
#!/bin/python3
import sys,time,re
from pprint import pprint
2024-12-12 20:02:54 +01:00
import numpy as np
2024-12-12 14:20:12 +01:00
sys.path.insert(0, '../../')
2024-12-12 17:55:31 +01:00
from fred import list2int,get_re,nprint,lprint,loadFile,toGrid,bfs,get_value_in_direction,addTuples,grid_valid
2024-12-12 14:20:12 +01:00
start_time = time.time()
2024-12-12 20:02:54 +01:00
input_f = 'input'
2024-12-12 14:20:12 +01:00
#########################################
# #
# Part 1 #
# #
#########################################
2024-12-13 21:42:53 +01:00
directions = {
'up': (-1, 0),
'down': (1, 0),
'left': (0, -1),
'right': (0, 1),
}
2024-12-12 14:20:12 +01:00
2024-12-13 21:42:53 +01:00
def get_neighbors(grid,node,visited):
2024-12-12 14:20:12 +01:00
neighbors = []
2024-12-13 21:42:53 +01:00
for d in directions.keys():
2024-12-12 14:20:12 +01:00
t = get_value_in_direction(grid,node)
if get_value_in_direction(grid,node,d) == t:
2024-12-13 21:42:53 +01:00
n = addTuples(directions[d],node)
2024-12-12 14:20:12 +01:00
if n not in visited:
neighbors.append(n)
visited.append(n)
neighbors+=get_neighbors(grid,n,visited)
return neighbors
def part1():
grid = toGrid(input_f)
visited = []
total_plots = []
for r,row in enumerate(grid):
for c,col in enumerate(row):
pos = (r,c)
plot = []
current = get_value_in_direction(grid,pos)
if pos not in visited:
x = get_neighbors(grid,pos,visited)
plot += x
if pos not in plot:
plot.append(pos)
total_plots.append(plot)
result = 0
for v in total_plots:
sides = 0
for x,y in v:
2024-12-13 21:42:53 +01:00
for dx, dy in directions.values():
2024-12-12 14:20:12 +01:00
neighbor = (x + dx, y + dy)
if neighbor in v:
sides += 1
total_sides = len(v) * 4 - sides
result += (total_sides*len(v))
return result
start_time = time.time()
2024-12-12 20:02:54 +01:00
print('Part 1:',part1(), '\t', round((time.time() - start_time)*1000), 'ms')
2024-12-12 14:20:12 +01:00
#########################################
# #
# Part 2 #
# #
#########################################
2024-12-12 17:55:31 +01:00
2024-12-12 14:20:12 +01:00
def part2():
grid = toGrid(input_f)
2024-12-12 20:02:54 +01:00
#print(grid)
2024-12-12 14:20:12 +01:00
#nprint(grid)
values = {}
visited = []
total_plots = []
2024-12-12 17:55:31 +01:00
2024-12-12 14:20:12 +01:00
for r,row in enumerate(grid):
for c,col in enumerate(row):
pos = (r,c)
plot = []
current = get_value_in_direction(grid,pos)
if pos not in visited:
2024-12-13 21:42:53 +01:00
x = get_neighbors(grid,pos,visited)
2024-12-12 14:20:12 +01:00
plot += x
if pos not in plot:
plot.append(pos)
if current not in values:
values[current] = []
total_plots.append(plot)
2024-12-12 17:55:31 +01:00
#pprint(total_plots)
2024-12-12 14:20:12 +01:00
2024-12-12 17:55:31 +01:00
for pdx,p in enumerate(total_plots):
total_plots[pdx] = sorted(p)
2024-12-12 20:02:54 +01:00
#pprint(total_plots)
2024-12-12 14:20:12 +01:00
2024-12-12 17:55:31 +01:00
#directions = [(0, 1), (0, -1), (1, 0), (-1, 0)]
directions = [(0, 1), (1, 0)]
#directions = [(1, 1), (1, -1), (-1, -1), (-1, 1),(0, 1), (0, -1), (1, 0), (-1, 0)]
2024-12-12 14:20:12 +01:00
2024-12-12 17:55:31 +01:00
result = 0
2024-12-12 14:20:12 +01:00
2024-12-12 17:55:31 +01:00
def find_edge(r,c,p,visited):
edge = True
end = ()
for dr, dc in directions:
neighbor = (r + dr, c + dc)
if neighbor in p:
if neighbor not in visited:
#print((r,c),neighbor)
visited.append((r,c))
find_edge((r + dr),(c + dc),p,visited)
else:
#if (r+0,c+1) not in p or (r+1,c+0) not in p:
edge = False
end = (r,c)
#print(r,c,(dr,dc))
#input()
if not edge:
print(end)
for p in total_plots:
2024-12-12 20:02:54 +01:00
area = len(p)
fences = 0
2024-12-12 17:55:31 +01:00
for r,c in p:
visited = []
2024-12-12 20:02:54 +01:00
#print('@',(r,c),get_value_in_direction(grid,(r,c)))
if get_value_in_direction(grid,(r,c),'up') != grid[r][c] and get_value_in_direction(grid,(r,c),'left') != grid[r][c]:
#print('┌ corner - up+left')
fences+=1
if get_value_in_direction(grid,(r,c),'up') == grid[r][c] and get_value_in_direction(grid,(r,c),'left') == grid[r][c] and get_value_in_direction(grid,(r,c),'up-left') != grid[r][c]:
#print('└ corner - up+left+upleft')
fences+=1
if get_value_in_direction(grid,(r,c),'left') != grid[r][c] and get_value_in_direction(grid,(r,c),'down') != grid[r][c]:
#print('└ corner - left+down')
fences+=1
if get_value_in_direction(grid,(r,c),'left') == grid[r][c] and get_value_in_direction(grid,(r,c),'down') == grid[r][c] and get_value_in_direction(grid,(r,c),'down-left') != grid[r][c]:
#print('┐ corner - left+down+downleft')
fences+=1
if get_value_in_direction(grid,(r,c),'down') != grid[r][c] and get_value_in_direction(grid,(r,c),'right') != grid[r][c]:
#print('┘ corner - down+right')
fences+=1
if get_value_in_direction(grid,(r,c),'down') == grid[r][c] and get_value_in_direction(grid,(r,c),'right') == grid[r][c] and get_value_in_direction(grid,(r,c),'down-right') != grid[r][c]:
#print('┌ corner - down+right+downright')
fences+=1
if get_value_in_direction(grid,(r,c),'right') != grid[r][c] and get_value_in_direction(grid,(r,c),'up') != grid[r][c]:
#print('┐ corner - right+up')
fences+=1
if get_value_in_direction(grid,(r,c),'right') == grid[r][c] and get_value_in_direction(grid,(r,c),'up') == grid[r][c] and get_value_in_direction(grid,(r,c),'up-right') != grid[r][c]:
#print('└ corner - right+up+upright')
fences+=1
#┌ ┐
#└ ┘
#input()
2024-12-12 17:55:31 +01:00
2024-12-12 20:02:54 +01:00
result+=(area*fences)
2024-12-12 17:55:31 +01:00
# for v in total_plots:
# total_corners = 0
# corners = 0
# for x,y in v:
# print('Checking',(x,y))
# keep = True
# for dx, dy in directions:
# neighbor = (x + dx, y + dy) # Instead of finding all the neighbors, check if
# if neighbor in v:
# keep = False
# #corners += 1
# else:
# corners += 1
# #if corners < 2:
# # total_corners += 1
# print(corners)
#input()
#if corners == 3:
# total_corners += 2
#print(v,total_corners)
#result += (total_corners*len(v))
2024-12-12 14:20:12 +01:00
return result
start_time = time.time()
print('Part 2:',part2(), '\t\t', round((time.time() - start_time)*1000), 'ms')