浅谈golang结构体偷懒初始化
- 作者: 蜗牛的倔强
- 来源: 51数据库
- 2021-08-24
运行一段程序,警告:
service/mysqlconfig.go:63::error: golang.guazi-corp.com/tools/ksql-runner/model.creatingmysqlmongodbrecord composite literal uses unkeyed fields (vet)
其中,composite literal uses unkeyed fields这个警告找了很久原因,最终发现是结构体初始化的问题,自己埋雷。
例如,结构体定义如下,
type a struct {
*b
}
使用下边无键字段初始化,就会警告:
vara := a{b} // b is of type *b
在自己看来b的类型是b*,但编译器却不认账。
正确初始化方法是:
a := a{b: b}
有人说是,这只是针对"go vet"中的一个错误的解决方法。
因为go vet默认情况下执行所有检查,如果任何标志显式设置为true,则仅运行那些测试。
相反,如果任何标志显式设置为false,则仅禁用那些测试。
因此-printf = true运行printf检查,-printf = false运行除printf检查之外的所有检查。
可以禁用这个检查,也可以修改代码,但是规范代码更比较合适。
虽然这只是一个警告,有人觉得无关紧要,但是当你几周或者几个月后在结构体中添加了一个字段,那么你使用unkeyed初始化的所有东西都会被破坏。显然,在一个项目中不希望遇到这种错误。
补充:golang中结构体的初始化方法(new方法)
准备工作:
定义结构体:student
import (
"fmt"
"reflect"
)
type student struct {
studentid string `json:"sid"`
studentname string `json:"sname"`
studentclass string `json:"class"`
studentteacher string `json:"class"`
}
测试代码如下:
func main() {
s0 := student{}
fmt.println(s0)
fmt.println(reflect.typeof(s0))
s1 := new(student)
fmt.println(s1)
fmt.println(reflect.typeof(s1))
s2 := &student{}
fmt.println(s2)
fmt.println(reflect.typeof(s2))
s3 := &student{"", "", "100", "200"}
fmt.println(s3)
fmt.println(reflect.typeof(s3))
s4:= &student{studentid: "100", studentname: "200"}
fmt.println(s4)
fmt.println(reflect.typeof(s4))
}
输出结果:
{ }
main.student
&{ }
*main.student
&{ }
*main.student
&{ 100 200}
*main.student
&{100 200 }
*main.student
注意:
除了s0以外,s1-s4变量全部为指向rect结构的指针(指针变量),因为使用了new()函数和&操作符
而s0的方法 s0 := student{} 表示的是一个student类型,两者是不一样的
在go语言中,未进行初始化的变量都会被初始化为该类型的零值,例如bool类型的零值为false, int类型的零值为0, string类型的零值为空字符串. 在go语言中没有构造函数的概念,对象的创建通常交由一个全局的创建函数来完成,以newxxx来命令,表示"构造函数":
func newstudent() {
return &student() //返回指针
}
用 new 分配内存 内建函数 new 本质上说跟其他语言中的同名函数功能一样:new(t) 分配了零值填充的 t 类型的内存空间,并且返回其地址,一个 *t 类型的值。
务必记得 make 仅适用于 map,slice 和 channel,并且返回的不是指针。应当用 new获得特定的指针。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持。如有错误或未考虑完全的地方,望不吝赐教。
