mirror of
				https://github.com/onyx-and-iris/aoc2024.git
				synced 2025-10-25 10:11:45 +00:00 
			
		
		
		
	add point subpackage
build ShortestPath as ordered map replace dijkstra in part two with a bfs
This commit is contained in:
		
							parent
							
								
									62665d4d10
								
							
						
					
					
						commit
						22b442171b
					
				
							
								
								
									
										1
									
								
								day-18/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								day-18/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | run.sh | ||||||
| @ -2,14 +2,14 @@ goos: linux | |||||||
| goarch: amd64 | goarch: amd64 | ||||||
| pkg: github.com/onyx-and-iris/aoc2024/day-18 | pkg: github.com/onyx-and-iris/aoc2024/day-18 | ||||||
| cpu: Intel(R) Core(TM) i7-8700K CPU @ 3.70GHz | cpu: Intel(R) Core(TM) i7-8700K CPU @ 3.70GHz | ||||||
| BenchmarkSolve-12    	       1	3840363786 ns/op | BenchmarkSolve-12    	       1	1410266797 ns/op | ||||||
| BenchmarkSolve-12    	       1	3847307489 ns/op | BenchmarkSolve-12    	       1	1422531597 ns/op | ||||||
| BenchmarkSolve-12    	       1	3793560090 ns/op | BenchmarkSolve-12    	       1	1422451797 ns/op | ||||||
| BenchmarkSolve-12    	       1	3804761992 ns/op | BenchmarkSolve-12    	       1	1404297097 ns/op | ||||||
| BenchmarkSolve-12    	       1	3811796394 ns/op | BenchmarkSolve-12    	       1	1391816300 ns/op | ||||||
| BenchmarkSolve-12    	       1	3782742297 ns/op | BenchmarkSolve-12    	       1	1416344400 ns/op | ||||||
| BenchmarkSolve-12    	       1	3799861901 ns/op | BenchmarkSolve-12    	       1	1399980801 ns/op | ||||||
| BenchmarkSolve-12    	       1	3784385312 ns/op | BenchmarkSolve-12    	       1	1391180300 ns/op | ||||||
| BenchmarkSolve-12    	       1	3780708522 ns/op | BenchmarkSolve-12    	       1	1390386701 ns/op | ||||||
| BenchmarkSolve-12    	       1	3811926561 ns/op | BenchmarkSolve-12    	       1	1399453200 ns/op | ||||||
| ok  	github.com/onyx-and-iris/aoc2024/day-18	38.067s | ok  	github.com/onyx-and-iris/aoc2024/day-18	14.058s | ||||||
|  | |||||||
| @ -2,6 +2,15 @@ module github.com/onyx-and-iris/aoc2024/day-18 | |||||||
| 
 | 
 | ||||||
| go 1.23.3 | go 1.23.3 | ||||||
| 
 | 
 | ||||||
| require github.com/sirupsen/logrus v1.9.3 | require ( | ||||||
|  | 	github.com/sirupsen/logrus v1.9.3 | ||||||
|  | 	github.com/wk8/go-ordered-map/v2 v2.1.8 | ||||||
|  | ) | ||||||
| 
 | 
 | ||||||
| require golang.org/x/sys v0.12.0 // indirect | require ( | ||||||
|  | 	github.com/bahlo/generic-list-go v0.2.0 // indirect | ||||||
|  | 	github.com/buger/jsonparser v1.1.1 // indirect | ||||||
|  | 	github.com/mailru/easyjson v0.7.7 // indirect | ||||||
|  | 	golang.org/x/sys v0.12.0 // indirect | ||||||
|  | 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||||
|  | ) | ||||||
|  | |||||||
| @ -1,16 +1,28 @@ | |||||||
|  | github.com/bahlo/generic-list-go v0.2.0 h1:5sz/EEAK+ls5wF+NeqDpk5+iNdMDXrh3z3nPnH1Wvgk= | ||||||
|  | github.com/bahlo/generic-list-go v0.2.0/go.mod h1:2KvAjgMlE5NNynlg/5iLrrCCZ2+5xWbdbCW3pNTGyYg= | ||||||
|  | github.com/buger/jsonparser v1.1.1 h1:2PnMjfWD7wBILjqQbt530v576A/cAbQvEW9gGIpYMUs= | ||||||
|  | github.com/buger/jsonparser v1.1.1/go.mod h1:6RYKKt7H4d4+iWqouImQ9R2FZql3VbhNgx27UK13J/0= | ||||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
|  | github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y= | ||||||
|  | github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0= | ||||||
|  | github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= | ||||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||||
| github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ= | github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ= | ||||||
| github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= | github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= | ||||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
| github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= |  | ||||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||||
|  | github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk= | ||||||
|  | github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= | ||||||
|  | github.com/wk8/go-ordered-map/v2 v2.1.8 h1:5h/BUHu93oj4gIdvHHHGsScSTMijfx5PeYkE/fJgbpc= | ||||||
|  | github.com/wk8/go-ordered-map/v2 v2.1.8/go.mod h1:5nJHM5DyteebpVlHnWMV0rPz6Zp7+xBAnxjb1X5vnTw= | ||||||
| golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.12.0 h1:CM0HF96J0hcLAwsHPJZjfdNzs0gftsLfgKt57wWHJ0o= | golang.org/x/sys v0.12.0 h1:CM0HF96J0hcLAwsHPJZjfdNzs0gftsLfgKt57wWHJ0o= | ||||||
| golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
|  | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= |  | ||||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||||
|  | gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||||
|  | gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||||
|  | |||||||
| @ -2,5 +2,5 @@ goos: linux | |||||||
| goarch: amd64 | goarch: amd64 | ||||||
| pkg: github.com/onyx-and-iris/aoc2024/day-18/internal/one | pkg: github.com/onyx-and-iris/aoc2024/day-18/internal/one | ||||||
| cpu: Intel(R) Core(TM) i7-8700K CPU @ 3.70GHz | cpu: Intel(R) Core(TM) i7-8700K CPU @ 3.70GHz | ||||||
| BenchmarkSolve-12    	1000000000	         0.004050 ns/op | BenchmarkSolve-12    	1000000000	         0.003678 ns/op | ||||||
| ok  	github.com/onyx-and-iris/aoc2024/day-18/internal/one	0.029s | ok  	github.com/onyx-and-iris/aoc2024/day-18/internal/one	0.030s | ||||||
|  | |||||||
| @ -3,11 +3,14 @@ package one | |||||||
| import ( | import ( | ||||||
| 	"slices" | 	"slices" | ||||||
| 	"strings" | 	"strings" | ||||||
|  | 
 | ||||||
|  | 	"github.com/onyx-and-iris/aoc2024/day-18/internal/point" | ||||||
|  | 	orderedmap "github.com/wk8/go-ordered-map/v2" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| type graph struct { | type graph struct { | ||||||
| 	start Point | 	start point.Point | ||||||
| 	end   Point | 	end   point.Point | ||||||
| 	data  []string | 	data  []string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -26,25 +29,26 @@ func newGraph(width, height, numCorruptions int, corruptedCoords [][]int) *graph | |||||||
| 		data[coords[1]] = replaceAtIndex(data[coords[1]], '#', coords[0]) | 		data[coords[1]] = replaceAtIndex(data[coords[1]], '#', coords[0]) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return &graph{Point{0, 0}, Point{len(data[0]) - 1, len(data) - 1}, data} | 	return &graph{point.Point{X: 0, Y: 0}, point.Point{X: len(data[0]) - 1, Y: len(data) - 1}, data} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *graph) String() string { | func (g *graph) String() string { | ||||||
| 	return strings.Join(g.data, "\n") | 	return strings.Join(g.data, "\n") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *graph) isOutOfBounds(p Point) bool { | func (g *graph) isOutOfBounds(p point.Point) bool { | ||||||
| 	return p.X < 0 || p.Y < 0 || p.Y >= len(g.data) || p.X >= len(g.data[p.Y]) | 	return p.X < 0 || p.Y < 0 || p.Y >= len(g.data) || p.X >= len(g.data[p.Y]) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *graph) valueAt(p Point) rune { | func (g *graph) valueAt(p point.Point) rune { | ||||||
| 	return rune(g.data[p.Y][p.X]) | 	return rune(g.data[p.Y][p.X]) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *graph) debug(path []Point) string { | func (g *graph) debug(path *orderedmap.OrderedMap[point.Point, struct{}]) string { | ||||||
| 	temp := slices.Clone(g.data) | 	temp := slices.Clone(g.data) | ||||||
| 	for _, p := range path { | 	temp[g.start.Y] = replaceAtIndex(temp[g.start.Y], 'O', g.start.X) | ||||||
| 		temp[p.Y] = replaceAtIndex(temp[p.Y], 'O', p.X) | 	for pair := path.Oldest(); pair != nil; pair = pair.Next() { | ||||||
|  | 		temp[pair.Key.Y] = replaceAtIndex(temp[pair.Key.Y], 'O', pair.Key.X) | ||||||
| 	} | 	} | ||||||
| 	return strings.Join(temp, "\n") | 	return strings.Join(temp, "\n") | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,10 +1,12 @@ | |||||||
| package one | package one | ||||||
| 
 | 
 | ||||||
| func neighbours(p Point) [4]Point { | import "github.com/onyx-and-iris/aoc2024/day-18/internal/point" | ||||||
| 	return [4]Point{ | 
 | ||||||
| 		{p.X, p.Y - 1}, // N | func neighbours(p point.Point) [4]point.Point { | ||||||
| 		{p.X + 1, p.Y}, // E | 	return [4]point.Point{ | ||||||
| 		{p.X, p.Y + 1}, // S | 		{X: p.X, Y: p.Y - 1}, // N | ||||||
| 		{p.X - 1, p.Y}, // W | 		{X: p.X + 1, Y: p.Y}, // E | ||||||
|  | 		{X: p.X, Y: p.Y + 1}, // S | ||||||
|  | 		{X: p.X - 1, Y: p.Y}, // W | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | |||||||
| @ -5,11 +5,14 @@ import ( | |||||||
| 	"math" | 	"math" | ||||||
| 
 | 
 | ||||||
| 	"github.com/onyx-and-iris/aoc2024/day-18/internal/config" | 	"github.com/onyx-and-iris/aoc2024/day-18/internal/config" | ||||||
|  | 	"github.com/onyx-and-iris/aoc2024/day-18/internal/point" | ||||||
| 	"github.com/onyx-and-iris/aoc2024/day-18/internal/queue" | 	"github.com/onyx-and-iris/aoc2024/day-18/internal/queue" | ||||||
| 	log "github.com/sirupsen/logrus" | 	log "github.com/sirupsen/logrus" | ||||||
|  | 
 | ||||||
|  | 	orderedmap "github.com/wk8/go-ordered-map/v2" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| var ShortestPath []Point | var ShortestPath *orderedmap.OrderedMap[point.Point, struct{}] | ||||||
| 
 | 
 | ||||||
| func Solve(buf []byte, config config.Config) (int, error) { | func Solve(buf []byte, config config.Config) (int, error) { | ||||||
| 	r := bytes.NewReader(buf) | 	r := bytes.NewReader(buf) | ||||||
| @ -21,11 +24,11 @@ func Solve(buf []byte, config config.Config) (int, error) { | |||||||
| 	log.Debugf("start: %v end: %v", graph.start, graph.end) | 	log.Debugf("start: %v end: %v", graph.start, graph.end) | ||||||
| 
 | 
 | ||||||
| 	log.Debugf("\n%s\n", graph.String()) | 	log.Debugf("\n%s\n", graph.String()) | ||||||
| 	queue := queue.New[Point]() | 	queue := queue.New[point.Point]() | ||||||
| 	queue.Enqueue(graph.start) | 	queue.Enqueue(graph.start) | ||||||
| 	visited := make(map[Point]struct{}) | 	visited := make(map[point.Point]struct{}) | ||||||
| 	costs := make(map[Point]int) | 	costs := make(map[point.Point]int) | ||||||
| 	prev := make(map[Point]Point) | 	prev := make(map[point.Point]point.Point) | ||||||
| 
 | 
 | ||||||
| 	for !queue.IsEmpty() { | 	for !queue.IsEmpty() { | ||||||
| 		current := queue.Dequeue() | 		current := queue.Dequeue() | ||||||
| @ -63,14 +66,15 @@ func Solve(buf []byte, config config.Config) (int, error) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ShortestPath = []Point{graph.end} | 	ShortestPath = orderedmap.New[point.Point, struct{}]() | ||||||
|  | 	ShortestPath.Set(graph.end, struct{}{}) | ||||||
| 	node := prev[graph.end] | 	node := prev[graph.end] | ||||||
| 	for node != graph.start { | 	for node != graph.start { | ||||||
| 		ShortestPath = append(ShortestPath, prev[node]) | 		ShortestPath.Set(node, struct{}{}) | ||||||
| 		node = prev[node] | 		node = prev[node] | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	log.Debugf("\n%s\n", graph.debug(ShortestPath)) | 	log.Debugf("\n%s\n", graph.debug(ShortestPath)) | ||||||
| 
 | 
 | ||||||
| 	return len(ShortestPath), nil | 	return ShortestPath.Len(), nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| package one | package point | ||||||
| 
 | 
 | ||||||
| type Point struct { | type Point struct { | ||||||
| 	X int | 	X int | ||||||
| @ -2,5 +2,5 @@ goos: linux | |||||||
| goarch: amd64 | goarch: amd64 | ||||||
| pkg: github.com/onyx-and-iris/aoc2024/day-18/internal/two | pkg: github.com/onyx-and-iris/aoc2024/day-18/internal/two | ||||||
| cpu: Intel(R) Core(TM) i7-8700K CPU @ 3.70GHz | cpu: Intel(R) Core(TM) i7-8700K CPU @ 3.70GHz | ||||||
| BenchmarkSolve-12    	       1	3756699083 ns/op | BenchmarkSolve-12    	       1	1414311557 ns/op | ||||||
| ok  	github.com/onyx-and-iris/aoc2024/day-18/internal/two	3.760s | ok  	github.com/onyx-and-iris/aoc2024/day-18/internal/two	1.418s | ||||||
|  | |||||||
| @ -1,17 +1,17 @@ | |||||||
| package two | package two | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"errors" |  | ||||||
| 	"math" |  | ||||||
| 	"slices" | 	"slices" | ||||||
| 	"strings" | 	"strings" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/onyx-and-iris/aoc2024/day-18/internal/point" | ||||||
| 	"github.com/onyx-and-iris/aoc2024/day-18/internal/queue" | 	"github.com/onyx-and-iris/aoc2024/day-18/internal/queue" | ||||||
|  | 	log "github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| type graph struct { | type graph struct { | ||||||
| 	start point | 	start point.Point | ||||||
| 	end   point | 	end   point.Point | ||||||
| 	data  []string | 	data  []string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -30,38 +30,35 @@ func newGraph(width, height, numCorruptions int, corruptedCoords [][]int) *graph | |||||||
| 		data[coords[1]] = replaceAtIndex(data[coords[1]], '#', coords[0]) | 		data[coords[1]] = replaceAtIndex(data[coords[1]], '#', coords[0]) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return &graph{point{0, 0}, point{len(data[0]) - 1, len(data) - 1}, data} | 	return &graph{point.Point{X: 0, Y: 0}, point.Point{X: len(data[0]) - 1, Y: len(data) - 1}, data} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *graph) String() string { | func (g *graph) String() string { | ||||||
| 	return strings.Join(g.data, "\n") | 	return strings.Join(g.data, "\n") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *graph) isOutOfBounds(p point) bool { | func (g *graph) isOutOfBounds(p point.Point) bool { | ||||||
| 	return p.x < 0 || p.y < 0 || p.y >= len(g.data) || p.x >= len(g.data[p.y]) | 	return p.X < 0 || p.Y < 0 || p.Y >= len(g.data) || p.X >= len(g.data[p.Y]) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *graph) valueAt(p point) rune { | func (g *graph) valueAt(p point.Point) rune { | ||||||
| 	return rune(g.data[p.y][p.x]) | 	return rune(g.data[p.Y][p.X]) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *graph) addCorruption(coords []int) { | func (g *graph) addCorruption(p point.Point) { | ||||||
| 	g.data[coords[1]] = replaceAtIndex(g.data[coords[1]], '#', coords[0]) | 	g.data[p.Y] = replaceAtIndex(g.data[p.Y], '#', p.X) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *graph) dijkstra(start, end point) ([]point, error) { | func (g *graph) bfs() bool { | ||||||
| 	queue := queue.New[point]() | 	queue := queue.New[point.Point]() | ||||||
| 	queue.Enqueue(start) | 	queue.Enqueue(g.start) | ||||||
| 	visited := make(map[point]struct{}) | 	visited := make(map[point.Point]struct{}) | ||||||
| 	costs := make(map[point]int) |  | ||||||
| 	prev := make(map[point]point) |  | ||||||
| 
 | 
 | ||||||
| 	for !queue.IsEmpty() { | 	for !queue.IsEmpty() { | ||||||
| 		current := queue.Dequeue() | 		current := queue.Dequeue() | ||||||
| 
 | 
 | ||||||
| 		// we found a shortest path | 		if current == g.end { | ||||||
| 		if current == end { | 			return true | ||||||
| 			return g.generatePath(start, end, prev), nil |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		_, ok := visited[current] | 		_, ok := visited[current] | ||||||
| @ -79,40 +76,21 @@ func (g *graph) dijkstra(start, end point) ([]point, error) { | |||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			_, ok := costs[n] |  | ||||||
| 			if !ok { |  | ||||||
| 				costs[n] = math.MaxInt |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			new_cost := costs[current] + 1 |  | ||||||
| 			if new_cost < costs[n] { |  | ||||||
| 				costs[n] = new_cost |  | ||||||
| 				prev[n] = current |  | ||||||
| 			queue.Enqueue(n) | 			queue.Enqueue(n) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 	log.Debugf("\n%s\n", g.debug(visited)) | ||||||
|  | 
 | ||||||
|  | 	return false | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 	return nil, errors.New("unable to find a shortest path") | func (g *graph) debug(visited map[point.Point]struct{}) string { | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (g *graph) generatePath(start, end point, prev map[point]point) []point { |  | ||||||
| 	path := []point{end} |  | ||||||
| 	node := prev[end] |  | ||||||
| 	for node != start { |  | ||||||
| 		path = append(path, prev[node]) |  | ||||||
| 		node = prev[node] |  | ||||||
| 	} |  | ||||||
| 	return path |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (g *graph) debug(path []point) string { |  | ||||||
| 	temp := slices.Clone(g.data) | 	temp := slices.Clone(g.data) | ||||||
| 	for _, p := range path { | 	for p := range visited { | ||||||
| 		if g.valueAt(p) == '#' { | 		if g.valueAt(p) == '#' { | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
| 		temp[p.y] = replaceAtIndex(temp[p.y], 'O', p.x) | 		temp[p.Y] = replaceAtIndex(temp[p.Y], 'O', p.X) | ||||||
| 	} | 	} | ||||||
| 	return strings.Join(temp, "\n") | 	return strings.Join(temp, "\n") | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,10 +1,12 @@ | |||||||
| package two | package two | ||||||
| 
 | 
 | ||||||
| func neighbours(p point) [4]point { | import "github.com/onyx-and-iris/aoc2024/day-18/internal/point" | ||||||
| 	return [4]point{ | 
 | ||||||
| 		{p.x, p.y - 1}, // N | func neighbours(p point.Point) [4]point.Point { | ||||||
| 		{p.x + 1, p.y}, // E | 	return [4]point.Point{ | ||||||
| 		{p.x, p.y + 1}, // S | 		{X: p.X, Y: p.Y - 1}, // N | ||||||
| 		{p.x - 1, p.y}, // W | 		{X: p.X + 1, Y: p.Y}, // E | ||||||
|  | 		{X: p.X, Y: p.Y + 1}, // S | ||||||
|  | 		{X: p.X - 1, Y: p.Y}, // W | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,6 +0,0 @@ | |||||||
| package two |  | ||||||
| 
 |  | ||||||
| type point struct { |  | ||||||
| 	x int |  | ||||||
| 	y int |  | ||||||
| } |  | ||||||
| @ -5,6 +5,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 
 | 
 | ||||||
| 	"github.com/onyx-and-iris/aoc2024/day-18/internal/config" | 	"github.com/onyx-and-iris/aoc2024/day-18/internal/config" | ||||||
|  | 	"github.com/onyx-and-iris/aoc2024/day-18/internal/point" | ||||||
| 	log "github.com/sirupsen/logrus" | 	log "github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -16,6 +17,7 @@ func Solve(buf []byte, config config.Config) (string, error) { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	log.Debugf("start: %v end: %v", graph.start, graph.end) | 	log.Debugf("start: %v end: %v", graph.start, graph.end) | ||||||
|  | 	log.Debug(corruptedCoords) | ||||||
| 
 | 
 | ||||||
| 	indx := runUntilNoPath(graph, corruptedCoords, config) | 	indx := runUntilNoPath(graph, corruptedCoords, config) | ||||||
| 	return fmt.Sprintf("%d,%d", corruptedCoords[indx][0], corruptedCoords[indx][1]), nil | 	return fmt.Sprintf("%d,%d", corruptedCoords[indx][0], corruptedCoords[indx][1]), nil | ||||||
| @ -23,17 +25,15 @@ func Solve(buf []byte, config config.Config) (string, error) { | |||||||
| 
 | 
 | ||||||
| func runUntilNoPath(graph *graph, corruptedCoords [][]int, config config.Config) int { | func runUntilNoPath(graph *graph, corruptedCoords [][]int, config config.Config) int { | ||||||
| 	for i, coords := range corruptedCoords[config.NumCorruptions+1:] { | 	for i, coords := range corruptedCoords[config.NumCorruptions+1:] { | ||||||
| 		nextCorruption := point{coords[0], coords[1]} | 		nextCorruption := point.Point{X: coords[0], Y: coords[1]} | ||||||
| 
 | 
 | ||||||
| 		log.Debugf("adding corruption %v", nextCorruption) | 		log.Debugf("adding corruption %v", nextCorruption) | ||||||
|  | 		graph.addCorruption(nextCorruption) | ||||||
| 
 | 
 | ||||||
| 		graph.addCorruption(coords) | 		ok := graph.bfs() | ||||||
| 		path, err := graph.dijkstra(graph.start, graph.end) | 		if !ok { | ||||||
| 		if err != nil { |  | ||||||
| 			log.Debug(err) |  | ||||||
| 			return config.NumCorruptions + i + 1 | 			return config.NumCorruptions + i + 1 | ||||||
| 		} | 		} | ||||||
| 		log.Debugf("\n%s\n", graph.debug(path)) |  | ||||||
| 	} | 	} | ||||||
| 	return 0 | 	return 0 | ||||||
| } | } | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user