To prawda, że powyższe przykłady użycia const
i iota
są najbardziej idiomatycznymi sposobami reprezentowania prymitywnych wyliczeń w Go. Ale co, jeśli szukasz sposobu na stworzenie w pełni funkcjonalnego wyliczenia podobnego do typu, który można zobaczyć w innym języku, takim jak Java lub Python?
Bardzo prostym sposobem na utworzenie obiektu, który zaczyna wyglądać i zachowywać się jak wyliczenie ciągu w Pythonie, byłoby:
package main
import (
"fmt"
)
var Colors = newColorRegistry()
func newColorRegistry() *colorRegistry {
return &colorRegistry{
Red: "red",
Green: "green",
Blue: "blue",
}
}
type colorRegistry struct {
Red string
Green string
Blue string
}
func main() {
fmt.Println(Colors.Red)
}
Załóżmy, że chciałeś także mieć kilka metod narzędziowych, takich jak Colors.List()
i Colors.Parse("red")
. A twoje kolory były bardziej złożone i musiały być strukturą. Następnie możesz zrobić coś takiego:
package main
import (
"errors"
"fmt"
)
var Colors = newColorRegistry()
type Color struct {
StringRepresentation string
Hex string
}
func (c *Color) String() string {
return c.StringRepresentation
}
func newColorRegistry() *colorRegistry {
red := &Color{"red", "F00"}
green := &Color{"green", "0F0"}
blue := &Color{"blue", "00F"}
return &colorRegistry{
Red: red,
Green: green,
Blue: blue,
colors: []*Color{red, green, blue},
}
}
type colorRegistry struct {
Red *Color
Green *Color
Blue *Color
colors []*Color
}
func (c *colorRegistry) List() []*Color {
return c.colors
}
func (c *colorRegistry) Parse(s string) (*Color, error) {
for _, color := range c.List() {
if color.String() == s {
return color, nil
}
}
return nil, errors.New("couldn't find it")
}
func main() {
fmt.Printf("%s\n", Colors.List())
}
W tym momencie na pewno działa, ale może ci się nie podobać powtarzalne definiowanie kolorów. Jeśli w tym momencie chcesz to wyeliminować, możesz użyć tagów na swojej strukturze i zrobić trochę wymyślnego zastanowienia, aby to skonfigurować, ale mam nadzieję, że to wystarczy, aby pokryć większość ludzi.