go: add «726. Number of Atoms»
Signed-off-by: Matej Focko <me@mfocko.xyz>
This commit is contained in:
parent
c69a2e96c7
commit
a718efd3d8
2 changed files with 120 additions and 0 deletions
92
go/number-of-atoms.go
Normal file
92
go/number-of-atoms.go
Normal file
|
@ -0,0 +1,92 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"slices"
|
||||||
|
"strconv"
|
||||||
|
"unicode"
|
||||||
|
)
|
||||||
|
|
||||||
|
func countOfAtoms(formula string) string {
|
||||||
|
var stack []map[string]int
|
||||||
|
stack = append(stack, make(map[string]int))
|
||||||
|
|
||||||
|
i := 0
|
||||||
|
for i < len(formula) {
|
||||||
|
if formula[i] == '(' {
|
||||||
|
stack = append(stack, make(map[string]int))
|
||||||
|
i++
|
||||||
|
} else if formula[i] == ')' {
|
||||||
|
count := stack[len(stack)-1]
|
||||||
|
stack = stack[:len(stack)-1]
|
||||||
|
|
||||||
|
i++
|
||||||
|
j := i
|
||||||
|
for j < len(formula) && unicode.IsDigit(rune(formula[j])) {
|
||||||
|
j++
|
||||||
|
}
|
||||||
|
|
||||||
|
multiplier := 1
|
||||||
|
if i < j {
|
||||||
|
parsedMultiplier, err := strconv.ParseInt(formula[i:j], 10, 0)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
multiplier = int(parsedMultiplier)
|
||||||
|
}
|
||||||
|
i = j
|
||||||
|
|
||||||
|
parent := stack[len(stack)-1]
|
||||||
|
for element, count := range count {
|
||||||
|
parent[element] += count * multiplier
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
j := i + 1
|
||||||
|
for j < len(formula) && unicode.IsLower(rune(formula[j])) {
|
||||||
|
j++
|
||||||
|
}
|
||||||
|
|
||||||
|
element := formula[i:j]
|
||||||
|
i = j
|
||||||
|
|
||||||
|
for j < len(formula) && unicode.IsDigit(rune(formula[j])) {
|
||||||
|
j++
|
||||||
|
}
|
||||||
|
|
||||||
|
count := 1
|
||||||
|
if i < j {
|
||||||
|
parsedCount, err := strconv.ParseInt(formula[i:j], 10, 0)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
count = int(parsedCount)
|
||||||
|
}
|
||||||
|
i = j
|
||||||
|
|
||||||
|
parent := stack[len(stack)-1]
|
||||||
|
parent[element] += count
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
finalCount := stack[0]
|
||||||
|
|
||||||
|
elements := make([]string, len(finalCount))
|
||||||
|
for element := range finalCount {
|
||||||
|
elements = append(elements, element)
|
||||||
|
}
|
||||||
|
slices.Sort(elements)
|
||||||
|
|
||||||
|
answer := ""
|
||||||
|
for _, element := range elements {
|
||||||
|
answer += element
|
||||||
|
|
||||||
|
count := finalCount[element]
|
||||||
|
if count > 1 {
|
||||||
|
answer += fmt.Sprintf("%d", count)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return answer
|
||||||
|
}
|
28
go/number-of-atoms_test.go
Normal file
28
go/number-of-atoms_test.go
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func _test_NumberOfAtoms(t *testing.T, input, expected string) {
|
||||||
|
result := countOfAtoms(input)
|
||||||
|
if result != expected {
|
||||||
|
t.Errorf("Expected ‹%s› for ‹%s›, but got: ‹%s›!", expected, input, result)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_NumberOfAtoms_Easy(t *testing.T) {
|
||||||
|
_test_NumberOfAtoms(t, "H2", "H2")
|
||||||
|
_test_NumberOfAtoms(t, "O2", "O2")
|
||||||
|
_test_NumberOfAtoms(t, "HO", "HO")
|
||||||
|
_test_NumberOfAtoms(t, "H2O", "H2O")
|
||||||
|
_test_NumberOfAtoms(t, "H2SO4", "H2O4S")
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_NumberOfAtoms_Medium(t *testing.T) {
|
||||||
|
_test_NumberOfAtoms(t, "Mg(OH)2", "H2MgO2")
|
||||||
|
_test_NumberOfAtoms(t, "C6H5OH", "C6H6O")
|
||||||
|
_test_NumberOfAtoms(t, "C6H5CH3", "C7H8")
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_NumberOfAtoms_Hard(t *testing.T) {
|
||||||
|
_test_NumberOfAtoms(t, "K4(ON(SO3)2)2", "K4N2O14S4")
|
||||||
|
}
|
Loading…
Reference in a new issue