以下示例使用iota定义从3333开始的一系列端口号。
package main
import (
"fmt"
)
const (
FirstPort = iota+3333
SecondPort
ThirdPort
)
func main() {
hostAndPort := "localhost:"+fmt.Sprint(SecondPort)
fmt.Printf("%s", hostAndPort )
// Output:
// localhost:3334
}
当组合主机名和端口时,我想避免在fmt.Sprint
中包装端口常量并简单地写入,例如,"localhost:"+SecondPort
。有没有办法使用iota将端口号定义为字符串常量,例如"3334"
?
以下不起作用:
FirstPort = string(iota + 3333)
也没有
FirstPort = fmt.Sprintf("%d", iota + 3333)
在constant declaration中,预先声明的标识符
iota
表示连续的无类型整数constants。
所以iota
为你提供整数常量。如果我们想要string
常数,我们需要找到一种方法将整数转换为其基数-10 string
表示。这种方式必须是constant expression,否则我们不能在常量声明中使用它。
对我们来说不幸的是,从整数到conversion的简单类型string
不会产生数值的基数10表示,但是:
将有符号或无符号整数值转换为字符串类型会生成包含整数的UTF-8表示形式的字符串。
因此结果将是一个持有单个符文的string
,其值(Unicode代码点)是源编号。
同样调用“转换器”函数,如strconv.Itoa()
或fmt.Sprint()
是不可能的,因为调用这些函数不能成为常量表达式的一部分,因此结果只能用于变量声明(更不用说我们不能使用iota
) ,它只允许在常量声明中)。
但仍有解决方案。
我认为这不值得麻烦和可读性的损失,但实际上你可以使用string
定义持有增加的十进制数的iota
常量。
该解决方案从数字构建“完整”数字。我们可以通过连接数字的数字(作为string
值)来获得base-10 string
表示。
要解决的最后一个问题是如何“列出”数字的数字。这是一个简单的算术:
i % 10
。i / 10 % 10
。i / 100 % 10
。要获得rune
的数字(在0..9
的范围内),我们可以简单地添加'0'
,并将其转换为string
。就这样。
这就是我们如何为1位数字符串编码编码:
n0 = string('0'+iota%10)
对于2位数字:
n00 = string('0'+iota/10%10) + string('0'+iota/1%10)
对于3位数字:
n000 = string('0'+iota/100%10) + string('0'+iota/10%10) + string('0'+iota/1%10)
让我们看看它的实际效果:
const (
P00 = string('0'+iota/10%10) + string('0'+iota/1%10)
P01
P02
P03
P04
P05
P06
P07
P08
P09
P10
P11
P12
P13
P14
P15
P16
P17
P18
P19
P20
)
打印结果:
fmt.Printf("%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n%v\n",
P00, P01, P02, P03, P04, P05, P06, P07, P08, P09,
P10, P11, P12, P13, P14, P15, P16, P17, P18, P19, P20)
输出(在Go Playground上试试):
00
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
到目前为止一切顺利,但我们如何从3333
开始呢?
也不是问题,可以轻松实现。我们可以简单地通过添加“初始”数字来移动iota。这就是全部。
让我们看一个第一个数字是3339
的例子:
const (
P3339 = string('0'+(iota+3339)/1000%10) +
string('0'+(iota+3339)/100%10) +
string('0'+(iota+3339)/10%10) +
string('0'+(iota+3339)/1%10)
P3340
P3341
)
func main() {
fmt.Println(P3339)
fmt.Println(P3340)
fmt.Println(P3341)
}
以上的输出是预期的(尝试在Go Playground):
3339
3340
3341