249 lines
5 KiB
Go
249 lines
5 KiB
Go
package d8
|
|
|
|
import (
|
|
"log"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
)
|
|
|
|
func difference(a []string, b []string) []string {
|
|
needle := a
|
|
haystack := b
|
|
|
|
if len(a) > len(b) {
|
|
needle = b
|
|
haystack = a
|
|
}
|
|
|
|
mb := make(map[string]struct{}, len(needle))
|
|
for _, x := range needle {
|
|
mb[x] = struct{}{}
|
|
}
|
|
var diff []string
|
|
for _, x := range haystack {
|
|
if _, found := mb[x]; !found {
|
|
diff = append(diff, x)
|
|
}
|
|
}
|
|
return diff
|
|
}
|
|
|
|
func merge(a []string, b []string) []string {
|
|
return append(append([]string{}, a...), b...)
|
|
}
|
|
|
|
func findSegment(digits [][]string, part []string) string {
|
|
for _, digit := range digits {
|
|
diff := difference(digit, part)
|
|
|
|
if len(digit) > len(part) && len(diff) == 1 {
|
|
return diff[0]
|
|
}
|
|
}
|
|
|
|
log.Fatal("can't find segment")
|
|
|
|
return ""
|
|
}
|
|
|
|
// aaaa
|
|
// b c
|
|
// b c
|
|
// dddd
|
|
// e f
|
|
// e f
|
|
// gggg
|
|
//
|
|
// a: 0, c: 1, f: 2, g: 3, e: 4, b: 5, d: 6
|
|
func findDisplay(digitsAsString string) []string {
|
|
display := make([]string, 7)
|
|
|
|
digits := [][]string{}
|
|
splitted := strings.Split(digitsAsString, " ")
|
|
sort.Slice(splitted, func(i, j int) bool { return len(splitted[i]) < len(splitted[j]) })
|
|
for _, digit := range splitted {
|
|
digits = append(digits, strings.Split(digit, ""))
|
|
}
|
|
|
|
// a
|
|
display[0] = findSegment(digits, digits[0])
|
|
// g
|
|
display[3] = findSegment(digits, merge(digits[2], []string{display[0]}))
|
|
// e
|
|
display[4] = findSegment(digits, merge(digits[2], []string{display[0], display[3]}))
|
|
// d
|
|
display[6] = findSegment(digits, merge(digits[1], []string{display[3]}))
|
|
// b
|
|
display[5] = findSegment(digits, merge(digits[1], []string{display[3], display[4], display[6]}))
|
|
// f
|
|
display[2] = findSegment(digits, []string{display[0], display[3], display[4], display[5], display[6]})
|
|
// c
|
|
display[1] = findSegment(digits, []string{display[0], display[2], display[3], display[4], display[5], display[6]})
|
|
|
|
return display
|
|
}
|
|
|
|
func isOne(digit []string, display []string) bool {
|
|
return len(digit) == 2
|
|
}
|
|
|
|
func isTwo(digit []string, display []string) bool {
|
|
|
|
if len(digit) != 5 {
|
|
return false
|
|
}
|
|
|
|
two := strings.Join([]string{display[0], display[1], display[3], display[4], display[6]}, "")
|
|
|
|
for _, segment := range digit {
|
|
if !strings.Contains(two, segment) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func isThree(digit []string, display []string) bool {
|
|
|
|
if len(digit) != 5 {
|
|
return false
|
|
}
|
|
|
|
three := strings.Join([]string{display[0], display[1], display[2], display[3], display[6]}, "")
|
|
|
|
for _, segment := range digit {
|
|
if !strings.Contains(three, segment) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func isFour(digit []string, display []string) bool {
|
|
return len(digit) == 4
|
|
}
|
|
|
|
func isFive(digit []string, display []string) bool {
|
|
|
|
if len(digit) != 5 {
|
|
return false
|
|
}
|
|
|
|
three := strings.Join([]string{display[0], display[2], display[3], display[5], display[6]}, "")
|
|
|
|
for _, segment := range digit {
|
|
if !strings.Contains(three, segment) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func isSix(digit []string, display []string) bool {
|
|
|
|
if len(digit) != 6 {
|
|
return false
|
|
}
|
|
|
|
three := strings.Join([]string{display[0], display[2], display[3], display[4], display[5], display[6]}, "")
|
|
|
|
for _, segment := range digit {
|
|
if !strings.Contains(three, segment) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func isSeven(digit []string, display []string) bool {
|
|
return len(digit) == 3
|
|
}
|
|
|
|
func isEight(digit []string, display []string) bool {
|
|
return len(digit) == 7
|
|
}
|
|
|
|
func isNine(digit []string, display []string) bool {
|
|
|
|
if len(digit) != 6 {
|
|
return false
|
|
}
|
|
|
|
three := strings.Join([]string{display[0], display[1], display[2], display[3], display[5], display[6]}, "")
|
|
|
|
for _, segment := range digit {
|
|
if !strings.Contains(three, segment) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func isZero(digit []string, display []string) bool {
|
|
|
|
if len(digit) != 6 {
|
|
return false
|
|
}
|
|
|
|
three := strings.Join([]string{display[0], display[1], display[2], display[3], display[4], display[5]}, "")
|
|
|
|
for _, segment := range digit {
|
|
if !strings.Contains(three, segment) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func digitToNumber(digit []string, display []string) int {
|
|
numberMap := map[int]func([]string, []string) bool{
|
|
0: isZero,
|
|
1: isOne,
|
|
2: isTwo,
|
|
3: isThree,
|
|
4: isFour,
|
|
5: isFive,
|
|
6: isSix,
|
|
7: isSeven,
|
|
8: isEight,
|
|
9: isNine,
|
|
}
|
|
|
|
for number, f := range numberMap {
|
|
if f(digit, display) {
|
|
return number
|
|
}
|
|
}
|
|
log.Fatal("wrong digit", digit, display)
|
|
return 0
|
|
}
|
|
|
|
func findCode(digitsAsString string, display []string) int {
|
|
numbers := []string{}
|
|
for _, digit := range strings.Split(digitsAsString, " ") {
|
|
numbers = append(numbers, strconv.Itoa(digitToNumber(strings.Split(digit, ""), display)))
|
|
}
|
|
asInt, _ := strconv.Atoi(strings.Join(numbers, ""))
|
|
|
|
return asInt
|
|
}
|
|
|
|
// TODO: refactor too complicated
|
|
func P2(in string) int {
|
|
lines := strings.Split(strings.TrimSpace(in), "\n")
|
|
sum := 0
|
|
for _, line := range lines {
|
|
wiringsAndCode := strings.Split(line, " | ")
|
|
display := findDisplay(wiringsAndCode[0])
|
|
sum += findCode(wiringsAndCode[1], display)
|
|
}
|
|
|
|
return sum
|
|
}
|