|
| 1 | +package main |
| 2 | + |
| 3 | +import ( |
| 4 | + "bufio" |
| 5 | + "os" |
| 6 | + "strconv" |
| 7 | + "strings" |
| 8 | +) |
| 9 | + |
| 10 | +type Range struct { |
| 11 | + start, stop, digits int |
| 12 | +} |
| 13 | + |
| 14 | +func (r Range) InvalidIDs(chunkLen int) map[int]struct{} { |
| 15 | + repeat := r.digits / chunkLen |
| 16 | + |
| 17 | + invalids := make(map[int]struct{}) |
| 18 | + |
| 19 | + for i := r.start / pow10(r.digits-chunkLen); i <= r.stop/(pow10(r.digits-chunkLen)+1); i++ { |
| 20 | + id := buildID(i, repeat) |
| 21 | + if id < r.start { |
| 22 | + continue |
| 23 | + } |
| 24 | + if id > r.stop { |
| 25 | + break |
| 26 | + } |
| 27 | + invalids[id] = struct{}{} |
| 28 | + } |
| 29 | + |
| 30 | + return invalids |
| 31 | +} |
| 32 | + |
| 33 | +func parseInput() []Range { |
| 34 | + scanner := bufio.NewScanner(os.Stdin) |
| 35 | + scanner.Scan() |
| 36 | + var list []Range |
| 37 | + for _, r := range strings.Split(scanner.Text(), ",") { |
| 38 | + limits := strings.Split(r, "-") |
| 39 | + start, _ := strconv.Atoi(limits[0]) |
| 40 | + stop, _ := strconv.Atoi(limits[1]) |
| 41 | + startDigits, stopDigits := digits(start), digits(stop) |
| 42 | + |
| 43 | + // split the range into two ranges |
| 44 | + // e.g., start->999 and 1000->stop |
| 45 | + // to work on single digits range later |
| 46 | + if startDigits == stopDigits { |
| 47 | + list = append(list, Range{start, stop, startDigits}) |
| 48 | + } else { |
| 49 | + list = append(list, Range{start, pow10(stopDigits-1) - 1, startDigits}) |
| 50 | + list = append(list, Range{pow10(startDigits), stop, stopDigits}) |
| 51 | + } |
| 52 | + } |
| 53 | + return list |
| 54 | +} |
| 55 | + |
| 56 | +// faster than |
| 57 | +// int(math.Pow10(exp)) |
| 58 | +func pow10(exp int) int { |
| 59 | + base := 10 |
| 60 | + result := 1 |
| 61 | + for { |
| 62 | + if exp&1 == 1 { |
| 63 | + result *= base |
| 64 | + } |
| 65 | + exp >>= 1 |
| 66 | + if exp == 0 { |
| 67 | + break |
| 68 | + } |
| 69 | + base *= base |
| 70 | + } |
| 71 | + return result |
| 72 | +} |
| 73 | + |
| 74 | +// faster than |
| 75 | +// len(strconv.Itoa(i)) |
| 76 | +func digits(i int) int { |
| 77 | + if i == 0 { |
| 78 | + return 1 |
| 79 | + } |
| 80 | + count := 0 |
| 81 | + for i != 0 { |
| 82 | + i /= 10 |
| 83 | + count++ |
| 84 | + } |
| 85 | + return count |
| 86 | +} |
| 87 | + |
| 88 | +// faster than |
| 89 | +// strconv.Atoi(strings.Repeat(strconv.Itoa(chunk), repeat)) |
| 90 | +func buildID(chunk, repeat int) int { |
| 91 | + n := chunk |
| 92 | + for i := 1; i < repeat; i++ { |
| 93 | + n = n*pow10(digits(chunk)) + chunk |
| 94 | + } |
| 95 | + return n |
| 96 | +} |
0 commit comments