我如何在 Go 中执行文字 *int64?
How do I do a literal *int64 in Go?
我有一个带有 *int64
字段的结构类型。
type SomeType struct {
SomeField *int64
}
在我的代码中的某个时刻,我想声明它的文字(比如,当我知道所说的值应该是 0,或者指向 0,你明白我的意思)
instance := SomeType{
SomeField: &0,
}
...除非这不起作用
./main.go:xx: cannot use &0 (type *int) as type *int64 in field value
所以我试试这个
instance := SomeType{
SomeField: &int64(0),
}
...但这也行不通
./main.go:xx: cannot take the address of int64(0)
我该怎么做?我能想到的唯一解决方案是使用占位符变量
var placeholder int64
placeholder = 0
instance := SomeType{
SomeField: &placeholder,
}
注意:当 &0
语法是 *int 而不是 *int64
. 时,fine 有效编辑:不,它没有。对此感到抱歉。
编辑:
显然我的问题含糊不清。我正在寻找一种方法 字面意思 一个 *int64
。这可以在构造函数内部使用,或者声明文字结构值,甚至可以作为其他函数的参数。但是辅助函数或使用不同的类型不是我正在寻找的解决方案。
Go 语言规范 (Address operators) 不允许获取数字常量的地址(不是 untyped 也不是 输入 常量)。
The operand must be addressable, that is, either a variable, pointer indirection, or slice indexing operation; or a field selector of an addressable struct operand; or an array indexing operation of an addressable array. As an exception to the addressability requirement, x
[in the expression of &x
] may also be a (possibly parenthesized) composite literal.
为什么不允许这样做,请参阅相关问题:. A similar question (similarly not allowed to take its address):
0) 通用解决方案(来自 Go 1.18)
Go 1.18 添加了泛型。这意味着我们可以创建一个单一的通用 Ptr()
函数,该函数 returns 指向我们传递给它的任何值的指针。希望它将被添加到标准库中。在此之前,您可以使用 github.com/icza/gog
, the gog.Ptr()
功能(披露:我是作者)。
这是它的样子:
func Ptr[T any](v T) *T {
return &v
}
正在测试:
i := Ptr(2)
log.Printf("%T %v", i, *i)
s := Ptr("abc")
log.Printf("%T %v", s, *s)
x := Ptr[any](nil)
log.Printf("%T %v", x, *x)
将输出(在 Go Playground 上尝试):
2009/11/10 23:00:00 *int 2
2009/11/10 23:00:00 *string abc
2009/11/10 23:00:00 *interface {} <nil>
你的其他选项(在 Go 1.18 之前)(在 Go Playground 上全部尝试):
1) 与 new()
您可以简单地使用内置 new()
函数分配一个新的零值 int64
并获取其地址:
instance := SomeType{
SomeField: new(int64),
}
但请注意,这只能用于分配和获取指向任何类型的零值的指针。
2) 带辅助变量
对于非零元素最简单和推荐的是使用一个可以获取地址的辅助变量:
helper := int64(2)
instance2 := SomeType{
SomeField: &helper,
}
3) 有辅助功能
注意: 我的 github.com/icza/gox
library, in the gox
包中提供了获取指向非零值指针的辅助函数,因此您不需要必须将这些添加到您需要的所有项目中。
或者如果你多次需要这个,你可以创建一个辅助函数来分配和 returns 一个 *int64
:
func create(x int64) *int64 {
return &x
}
并使用它:
instance3 := SomeType{
SomeField: create(3),
}
请注意,我们实际上没有分配任何东西,Go 编译器在我们返回函数参数的地址时这样做了。 Go 编译器执行逃逸分析,如果局部变量可能逃逸函数,则在堆上(而不是堆栈)分配局部变量。有关详细信息,请参阅
4) 单行匿名函数
instance4 := SomeType{
SomeField: func() *int64 { i := int64(4); return &i }(),
}
或作为(更短的)替代方案:
instance4 := SomeType{
SomeField: func(i int64) *int64 { return &i }(4),
}
5) 带分片字面量,索引取地址
如果您希望 *SomeField
不同于 0
,那么您需要一些可寻址的东西。
你仍然可以这样做,但那很丑陋:
instance5 := SomeType{
SomeField: &[]int64{5}[0],
}
fmt.Println(*instance2.SomeField) // Prints 5
这里发生的是 []int64
切片是用文字创建的,具有一个元素 (5
)。并且它被索引(第 0 个元素)并获取第 0 个元素的地址。在后台,还将分配一个 [1]int64
的数组并将其用作切片的后备数组。所以这里有很多样板。
6) 使用辅助结构文字
让我们检查一下可寻址要求的例外情况:
As an exception to the addressability requirement, x
[in the expression of &x
] may also be a (possibly parenthesized) composite literal.
这意味着获取复合文字的地址,例如结构文字是可以的。如果这样做,我们将分配结构值并获得指向它的指针。但如果是这样,另一个要求将对我们可用:“可寻址结构操作数的字段选择器”。所以如果struct literal包含一个int64
类型的字段,我们也可以取那个字段的地址!
让我们看看这个选项的实际效果。我们将使用这个包装器结构类型:
type intwrapper struct {
x int64
}
现在我们可以做:
instance6 := SomeType{
SomeField: &(&intwrapper{6}).x,
}
注意这个
&(&intwrapper{6}).x
表示如下:
& ( (&intwrapper{6}).x )
但是我们可以省略“外部”括号,因为地址运算符 &
应用于 selector expression.
的结果
另请注意,在后台会发生以下情况(这也是一个有效的语法):
&(*(&intwrapper{6})).x
7) 使用辅助匿名结构文字
原理与案例 #6 相同,但我们也可以使用匿名结构文字,因此不需要 helper/wrapper 结构类型定义:
instance7 := SomeType{
SomeField: &(&struct{ x int64 }{7}).x,
}
使用return一个int64变量地址的函数来解决问题。
In the below code we use function f
which accepts an integer and
returns a pointer value which holds the address of the integer. By using this method we can easily solve the above problem.
type myStr struct {
url *int64
}
func main() {
f := func(s int64) *int64 {
return &s
}
myStr{
url: f(12345),
}
}
还有另一种优雅的方法可以实现这一点,它不会产生太多样板代码,而且在我看来也不难看。如果我需要一个带有指向基元而不是值的指针的结构,以确保不会在整个项目中使用零值结构成员,我将创建一个将这些基元作为参数的函数。
您可以定义一个函数来创建您的结构,然后将基元传递给该函数,然后使用指向函数参数的指针。
type Config struct {
Code *uint8
Name *string
}
func NewConfig(code uint8, name string) *Config {
return &Config{
Code: &code,
Name: &name,
}
}
func UseConfig() {
config := NewConfig(1, "test")
// ...
}
// in case there are many values, modern IDE will highlight argument names for you, so you don't have to remember
func UseConfig2() {
config := NewConfig(
1,
"test",
)
// ...
}
如果您不介意使用第三方库,可以使用 lo 包,它使用具有 .ToPtr()
函数
的泛型 (go 1.18+)
ptr := lo.ToPtr("hello world")
// *string{"hello world"}
我有一个带有 *int64
字段的结构类型。
type SomeType struct {
SomeField *int64
}
在我的代码中的某个时刻,我想声明它的文字(比如,当我知道所说的值应该是 0,或者指向 0,你明白我的意思)
instance := SomeType{
SomeField: &0,
}
...除非这不起作用
./main.go:xx: cannot use &0 (type *int) as type *int64 in field value
所以我试试这个
instance := SomeType{
SomeField: &int64(0),
}
...但这也行不通
./main.go:xx: cannot take the address of int64(0)
我该怎么做?我能想到的唯一解决方案是使用占位符变量
var placeholder int64
placeholder = 0
instance := SomeType{
SomeField: &placeholder,
}
注意:当 时,fine 有效编辑:不,它没有。对此感到抱歉。&0
语法是 *int 而不是 *int64
.
编辑:
显然我的问题含糊不清。我正在寻找一种方法 字面意思 一个 *int64
。这可以在构造函数内部使用,或者声明文字结构值,甚至可以作为其他函数的参数。但是辅助函数或使用不同的类型不是我正在寻找的解决方案。
Go 语言规范 (Address operators) 不允许获取数字常量的地址(不是 untyped 也不是 输入 常量)。
The operand must be addressable, that is, either a variable, pointer indirection, or slice indexing operation; or a field selector of an addressable struct operand; or an array indexing operation of an addressable array. As an exception to the addressability requirement,
x
[in the expression of&x
] may also be a (possibly parenthesized) composite literal.
为什么不允许这样做,请参阅相关问题:
0) 通用解决方案(来自 Go 1.18)
Go 1.18 添加了泛型。这意味着我们可以创建一个单一的通用 Ptr()
函数,该函数 returns 指向我们传递给它的任何值的指针。希望它将被添加到标准库中。在此之前,您可以使用 github.com/icza/gog
, the gog.Ptr()
功能(披露:我是作者)。
这是它的样子:
func Ptr[T any](v T) *T {
return &v
}
正在测试:
i := Ptr(2)
log.Printf("%T %v", i, *i)
s := Ptr("abc")
log.Printf("%T %v", s, *s)
x := Ptr[any](nil)
log.Printf("%T %v", x, *x)
将输出(在 Go Playground 上尝试):
2009/11/10 23:00:00 *int 2
2009/11/10 23:00:00 *string abc
2009/11/10 23:00:00 *interface {} <nil>
你的其他选项(在 Go 1.18 之前)(在 Go Playground 上全部尝试):
1) 与 new()
您可以简单地使用内置 new()
函数分配一个新的零值 int64
并获取其地址:
instance := SomeType{
SomeField: new(int64),
}
但请注意,这只能用于分配和获取指向任何类型的零值的指针。
2) 带辅助变量
对于非零元素最简单和推荐的是使用一个可以获取地址的辅助变量:
helper := int64(2)
instance2 := SomeType{
SomeField: &helper,
}
3) 有辅助功能
注意: 我的 github.com/icza/gox
library, in the gox
包中提供了获取指向非零值指针的辅助函数,因此您不需要必须将这些添加到您需要的所有项目中。
或者如果你多次需要这个,你可以创建一个辅助函数来分配和 returns 一个 *int64
:
func create(x int64) *int64 {
return &x
}
并使用它:
instance3 := SomeType{
SomeField: create(3),
}
请注意,我们实际上没有分配任何东西,Go 编译器在我们返回函数参数的地址时这样做了。 Go 编译器执行逃逸分析,如果局部变量可能逃逸函数,则在堆上(而不是堆栈)分配局部变量。有关详细信息,请参阅
4) 单行匿名函数
instance4 := SomeType{
SomeField: func() *int64 { i := int64(4); return &i }(),
}
或作为(更短的)替代方案:
instance4 := SomeType{
SomeField: func(i int64) *int64 { return &i }(4),
}
5) 带分片字面量,索引取地址
如果您希望 *SomeField
不同于 0
,那么您需要一些可寻址的东西。
你仍然可以这样做,但那很丑陋:
instance5 := SomeType{
SomeField: &[]int64{5}[0],
}
fmt.Println(*instance2.SomeField) // Prints 5
这里发生的是 []int64
切片是用文字创建的,具有一个元素 (5
)。并且它被索引(第 0 个元素)并获取第 0 个元素的地址。在后台,还将分配一个 [1]int64
的数组并将其用作切片的后备数组。所以这里有很多样板。
6) 使用辅助结构文字
让我们检查一下可寻址要求的例外情况:
As an exception to the addressability requirement,
x
[in the expression of&x
] may also be a (possibly parenthesized) composite literal.
这意味着获取复合文字的地址,例如结构文字是可以的。如果这样做,我们将分配结构值并获得指向它的指针。但如果是这样,另一个要求将对我们可用:“可寻址结构操作数的字段选择器”。所以如果struct literal包含一个int64
类型的字段,我们也可以取那个字段的地址!
让我们看看这个选项的实际效果。我们将使用这个包装器结构类型:
type intwrapper struct {
x int64
}
现在我们可以做:
instance6 := SomeType{
SomeField: &(&intwrapper{6}).x,
}
注意这个
&(&intwrapper{6}).x
表示如下:
& ( (&intwrapper{6}).x )
但是我们可以省略“外部”括号,因为地址运算符 &
应用于 selector expression.
另请注意,在后台会发生以下情况(这也是一个有效的语法):
&(*(&intwrapper{6})).x
7) 使用辅助匿名结构文字
原理与案例 #6 相同,但我们也可以使用匿名结构文字,因此不需要 helper/wrapper 结构类型定义:
instance7 := SomeType{
SomeField: &(&struct{ x int64 }{7}).x,
}
使用return一个int64变量地址的函数来解决问题。
In the below code we use function
f
which accepts an integer and returns a pointer value which holds the address of the integer. By using this method we can easily solve the above problem.
type myStr struct {
url *int64
}
func main() {
f := func(s int64) *int64 {
return &s
}
myStr{
url: f(12345),
}
}
还有另一种优雅的方法可以实现这一点,它不会产生太多样板代码,而且在我看来也不难看。如果我需要一个带有指向基元而不是值的指针的结构,以确保不会在整个项目中使用零值结构成员,我将创建一个将这些基元作为参数的函数。
您可以定义一个函数来创建您的结构,然后将基元传递给该函数,然后使用指向函数参数的指针。
type Config struct {
Code *uint8
Name *string
}
func NewConfig(code uint8, name string) *Config {
return &Config{
Code: &code,
Name: &name,
}
}
func UseConfig() {
config := NewConfig(1, "test")
// ...
}
// in case there are many values, modern IDE will highlight argument names for you, so you don't have to remember
func UseConfig2() {
config := NewConfig(
1,
"test",
)
// ...
}
如果您不介意使用第三方库,可以使用 lo 包,它使用具有 .ToPtr()
函数
ptr := lo.ToPtr("hello world")
// *string{"hello world"}