前言
枚举类型是一种常用的数据类型,用于表示一组有限的、预定义的、具名的常量值。在枚举类型中,每个常量都是一个枚举值,它们之间的值相等且唯一。
枚举类型通常用于表示一组相关的常量,比如星期、月份、性别等等。在其他语言里(比如 Java
和 C
),都内置了枚举类型,而在 Go
语言里是没有内置枚举类型的,因此我们需要采用其他方式实现类似的枚举类型功能,本文将介绍如何实现 “枚举类型”。
Go 语言中的 “枚举类型”
枚举类型的值本质上是常量,因此我们可以使用 Go
语言中的常量来实现类似枚举类型的功能,例如:
const (
Sunday = 1
Tuesday = 2
Wednesday = 3
Thursday = 4
Friday = 5
Saturday = 6
Monday = 7
)
复制代码
在这个例子中,我们使用了 const
关键字定义了一组常量,其中每个常量的名称代表着一个枚举,其值为对应的整数。
虽然这个例子能实现类似的枚举类型,但它不具备枚举类型的所有特征,例如缺少安全性和约束性,为了解决这两个问题,我们可以使用自定义类型进行改进:
type WeekDay int
const (
Sunday WeekDay = 1
Tuesday WeekDay = 2
Wednesday WeekDay = 3
Thursday WeekDay = 4
Friday WeekDay = 5
Saturday WeekDay = 6
Monday WeekDay = 7
)
复制代码
在改进后的例子中,我们定义了一个自定义类型 Weekday
,用于表示星期几。使用 const
关键字定义一个常量组,其中每个常量都被赋予了一个具体的值,同时使用 Weekday
类型进行类型约束和类型检查。这样,我们就可以通过枚举值的名称来表示某个特定的星期几,并且由于使用了自定义类型,编译器可以进行类型检查,从而提高了类型安全性。
使用 iota 优雅实现枚举
通过前面的例子不难发现,当我们需要定义多个枚举值时,手动指定每个枚举常量的值会变得十分麻烦。为了解决这个问题,我们可以使用 iota
常量生成器,它可以帮助我们生成连续的整数值。
例如,使用 iota
定义一个星期几的枚举类型:
type WeekDay int
const (
Sunday WeekDay = iota
Tuesday
Wednesday
Thursday
Friday
Saturday
Monday
)
复制代码
在这个例子中,我们使用 iota
自增常量生成器来定义了一个星期几的枚举类型,每个枚举值都是一个 Weekday
类型的常量。由于 iota
的自增规则,每个枚举值的值将自动递增,从而生成一系列连续的整数值。
为自定义的枚举添加方法
type WeekDay int
const (
Sunday WeekDay = iota
Tuesday
Wednesday
Thursday
Friday
Saturday
Monday
)
复制代码
为了能让我们实现的 “枚举类型” 更加具备枚举类型的特征,我们可以为其添加类似 Java
等其他语言中的枚举方法。
Name()
返回枚举值的名称。
// Name 返回枚举值的名称
func (w WeekDay) Name() string {
if w < Sunday || w > Monday {
return "Unknown"
}
return [...]string{"Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"}[w]
}
复制代码
Original
返回枚举值在枚举类型中的位置。
// Original 由于这个枚举类型的枚举值本质上是整数常量,因此我们可以直接使用枚举值作为它的序号
func (w WeekDay) Original() int {
return int(w)
}
复制代码
String()
实现 String
方法,用于打印输出。
// 将枚举值转成字符串,便于输出
func (w WeekDay) String() string {
return w.Name()
}
复制代码
Values()
返回一个包含所有枚举值的切片。
// Values 返回一个包含所有枚举值的切片
func Values() []WeekDay {
return []WeekDay{Monday, Tuesday, Wednesday, Thursday, Friday, Saturday, Sunday}
}
复制代码
ValueOf()
根据名称返回对应的枚举值。
// ValueOf 使用 switch 语句来根据名称返回对应的枚举值
func ValueOf(name string) (WeekDay, error) {
switch name {
case "Sunday":
return Sunday, nil
case "Monday":
return Monday, nil
case "Tuesday":
return Tuesday, nil
case "Wednesday":
return Wednesday, nil
case "Thursday":
return Thursday, nil
case "Friday":
return Friday, nil
case "Saturday":
return Saturday, nil
default:
return 0, fmt.Errorf("invalid WeekDay name: %s", name)
}
}
复制代码
小结
在日常开发中,枚举类型是很常用的,虽然 Go 语言中没有内置枚举类型,但也不妨碍我们自己实现一个类似的 “枚举类型”。在实现的时候,需要考虑类型约束和安全性的问题。
实现类似枚举类型功能的方式有很多种,本文只是介绍了使用自定义类型的方式,如果理解了核心思想,我们还可以使用结构体等方式来实现类似枚举类型的功能。