-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathd9.go
125 lines (115 loc) · 2.07 KB
/
d9.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
package main
import (
"strconv"
)
func (*methods) D9P1(input string) string {
var drive []int
for _, c := range input {
f, err := strconv.Atoi(string(c))
if err != nil {
break
}
drive = append(drive, f)
}
var sum int64
var li, ri int // lef and right indices
var lf, rf int // left and right files
var id int // file ID
var pos int // current block position
var tomove int // how much of rf is left to move to lf
var free int // how much free space is left
ri = len(drive) - 1
for {
if li == ri {
sum += chksum(id, pos, tomove)
break
}
lf = drive[li]
if li%2 == 0 {
// holy moly that's a FILE
id = li / 2
sum += chksum(id, pos, lf)
pos += lf
li++
continue
}
// Space space wanna go to space yes please space. Space space. Go to space.
rf = drive[ri]
id = ri / 2
if tomove == 0 {
tomove = rf
}
if free == 0 {
free = lf
}
if tomove <= free {
free -= tomove
sum += chksum(id, pos, tomove)
pos += tomove
tomove = 0
ri -= 2
} else {
tomove -= free
sum += chksum(id, pos, free)
pos += free
free = 0
}
if free == 0 {
li++
}
}
return strconv.FormatInt(sum, 10)
}
type d9entity struct {
id int
pos int
size int
}
func (*methods) D9P2(input string) string {
var files []*d9entity
var space []*d9entity
var pos int
for i, c := range input {
f, err := strconv.Atoi(string(c))
if err != nil {
break
}
e := d9entity{
pos: pos,
size: f,
}
if i%2 == 0 {
e.id = i / 2
files = append(files, &e)
} else {
space = append(space, &e)
}
pos += f
}
for i := len(files) - 1; i >= 0; i-- {
f := files[i]
for j := 0; j < len(space); j++ {
s := space[j]
if s.pos > f.pos {
break
}
if s.size >= f.size {
s.size -= f.size
f.pos = s.pos
s.pos += f.size
break
}
}
}
var sum int64
for _, f := range files {
sum += chksum(f.id, f.pos, f.size)
}
return strconv.FormatInt(sum, 10)
}
func chksum(id, spos, size int) (sum int64) {
for pos := spos; pos < spos+size; pos++ {
sum += int64(pos) * int64(id)
}
return
}