Le langage Go prend en charge struct ; struct
, la traduction chinoise est appelée structure
, qui, comme les tableaux, est un type composite, pas un type de référence. [Recommandations associées : Tutoriel vidéo Go] struct
,中文翻译称为结构体
,与数组一样,属于复合类型,并非引用类型。【相关推荐:Go视频教程】
Go语言的struct,与C语言中的struct或其他面向对象编程语言中的类(class)类似,可以定义字段(属性)和方法,但也有很不同的地方,需要深入学习,才能区分他们之间的区别。
注意复合类型与引用类型之间的区别,这应该也是值传递和引用传递的区别吧。
使用struct关键字可以定义一个结构体,结构体中的成员,称为结构体的字段或属性。
type Member struct { id int name, email string gender, age int }
上面的代码中,我们定义了一个包含5个字段的结构体,可以看到,相同类型name
和email
、gender
和age
在同一行中定义,但比较好的编程习惯是每一行只定义一个字段,如:
type Member struct { id int name string email string gender int age int }
当然,结构体也可以不包含任何字段,称为空结构体
,struct{}表示一个空的结构体,注意,直接定义一个空的结构体并没有意义,但在并发编程中,channel之间的通讯,可以使用一个struct{}作为信号量。
ch := make(chan struct{}) ch <- struct{}{}
上面的例子中,我们定义了Member结构体类型,接下就可以这个自定义的类型创建变量了。
直接定义变量,这个使用方式并没有为字段赋初始值,因此所有字段都会被自动赋予自已类型的零值,比如
name
的值为空字符串"",age的值为0。
var m1 Member//所有字段均为空值
使用字面量创建变量,这种使用方式,可以在大括号中为结构体的成员赋初始值,有两种赋初始值的方式,一种是按字段在结构体中的顺序赋值,下面代码中
m2
就是使用这种方式,这种方式要求所有的字段都必须赋值,因此如果字段太多,每个字段都要赋值,会很繁琐,另一种则使用字段名为指定字段赋值,如下面代码中变量m3
的创建,使用这种方式,对于其他没有指定的字段,则使用该字段类型的零值作为初始化值。
var m2 = Member{1,"小明","xiaoming@163.com",1,18} // 简短变量声明方式:m2 := Member{1,"小明","xiaoming@163.com",1,18} var m3 = Member{id:2,"name":"小红"}// 简短变量声明方式:m3 := Member{id:2,"name":"小红"}
通过变量名,使用逗号(.)
,可以访问结构体类型中的字段,或为字段赋值,也可以对字段进行取址(&)操作。
fmt.Println(m2.name)//输出:小明 m3.name = "小花" fmt.Println(m3.name)//输出:小花 age := &m3.age *age = 20 fmt.Println(m3.age)//20
结构体与数组一样,都是值传递,比如当把数组或结构体作为实参传给函数的形参时,会复制一个副本,所以为了提高性能,一般不会把数组直接传递给函数,而是使用切片(引用类型)代替,而把结构体传给函数时,可以使用指针结构体
。
指针结构体,即一个指向结构体的指针,声明结构体变量时,在结构体类型前加*号,便声明一个指向结构体的指针,如:
注意,指针类型为引用类型,声明结构体指针时,如果未初始化,则初始值为nil,只有初始化后,才能访问字段或为字段赋值。
var m1 *Member m1.name = "小明"//错误用法,未初始化,m1为nil m1 = &Member{} m1.name = "小明"//初始化后,结构体指针指向某个结构体地址,才能访问字段,为字段赋值。
另外,使用Go内置new()函数,可以分配内存来初始化结构休,并返回分配的内存指针,因为已经初始化了,所以可以直接访问字段。
var m2 = new(Member) m2.name = "小红"
我们知道,如果将结构体转给函数,只是复制结构体的副本,如果在函数内修改结构体字段值,外面的结构体并不会受影响,而如果将结构体指针传给函数,则在函数中使用指针对结构体所做的修改,都会影响到指针指向的结构体。
func main() { m1 := Member{} m2 := new(Member) Change(m1,m2) fmt.Println(m1,m2) } func Change(m1 Member,m2 *Member){ m1.Name = "小明" m2.Name = "小红" }
上面的例子中,我们定义结构体字段名首字母是小写的,这意味着这些字段在包外不可见
Faites attention à la différence entre les types composites et les types de référence. Cela devrait également être la différence entre le passage de valeur et le passage de référence.
package member type Member struct { id int name string email string gender int age int } package main fun main(){ var m = member.Member{1,"小明","xiaoming@163.com",1,18}//会引发panic错误 }
Dans le code ci-dessus, nous définissons une structure contenant 5 champs. Vous pouvez voir que les mêmes types nom
et email
, genre</ code> et. <code>age
sont définis dans la même ligne, mais une meilleure habitude de programmation est de définir un seul champ dans chaque ligne, tel que :
type Member struct { Id int Name string Email string Gender int Age int }
type Member struct { Id int `json:"id,-"` Name string `json:"name"` Email string `json:"email"` Gender int `json:"gender,"` Age int `json:"age"` }
🎜Définissez directement les variables. Cette méthode d'utilisation n'attribue pas de valeur initiale au champ, donc tous les champs se verront automatiquement attribuer une valeur zéro de leur propre type. Par exemple, la valeur dename</. code> est une chaîne vide "" , la valeur de age est 0. 🎜</blockquote><div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:js;toolbar:false;">Id int `json:"id" gorm:"AUTO_INCREMENT"`</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div><blockquote>🎜Utilisez des littéraux pour créer des variables De cette façon, vous pouvez attribuer des valeurs initiales aux membres de la structure entre accolades. La première consiste à attribuer des valeurs initiales. valeurs par champ. Affectation séquentielle dans la structure, <code>m2
utilise cette méthode dans le code suivant. Cette méthode nécessite que tous les champs se voient attribuer une valeur, donc s'il y a trop de champs, chaque champ. doit se voir attribuer une valeur, qui sera très L'autre méthode consiste à utiliser le nom du champ pour spécifier l'affectation du champ, comme la création de la variablem3
dans le code suivant En utilisant cette méthode, pour. Pour d'autres champs non spécifiés, utilisez la valeur zéro du type de champ comme valeur d'initialisation. 🎜
func setName(m Member,name string){//普通函数 m.Name = name } func (m Member)setName(name string){//绑定到Member结构体的方法 m.Name = name }
(.)
, ou attribuez une valeur à un champ, ou effectuez une opération d'adresse (&) sur un champ. 🎜m := Member{} m.setName("小明") fmt.Println(m.Name)//输出为空
. structure du pointeur
. 🎜🎜Structure de pointeur, c'est-à-dire un pointeur vers une structure. Lors de la déclaration d'une variable de structure, ajoutez un signe * avant le type de structure pour déclarer un pointeur vers la structure, tel que : 🎜🎜Notez que le type de pointeur est. Type de référence, lors de la déclaration d'un pointeur de structure, s'il n'est pas initialisé, la valeur initiale est nulle Ce n'est qu'après l'initialisation que l'on peut accéder au champ ou lui attribuer une valeur. 🎜
func (m *Member)setName(name string){/将Member改为*Member m.Name = name } m := Member{} m.setName("小明") fmt.Println(m.Name)//小明
type Animal struct { Name string //名称 Color string //颜色 Height float32 //身高 Weight float32 //体重 Age int //年龄 } //奔跑 func (a Animal)Run() { fmt.Println(a.Name + "is running") } //吃东西 func (a Animal)Eat() { fmt.Println(a.Name + "is eating") } type Cat struct { a Animal } func main() { var c = Cat{ a: Animal{ Name: "猫猫", Color: "橙色", Weight: 10, Height: 30, Age: 5, }, } fmt.Println(c.a.Name) c.a.Run() }
type Lion struct { Animal //匿名字段 } func main(){ var lion = Lion{ Animal{ Name: "小狮子", Color: "灰色", }, } lion.Run() fmt.Println(lion.Name) }
, et n'est donc pas accessible dans d'autres packages, seul l'accès au sein du package est autorisé. 🎜🎜Dans l'exemple suivant, nous déclarons Member dans le package member, puis créons une variable dans le package principal. Cependant, comme les champs de la structure ne sont pas visibles en dehors du package, nous ne pouvons pas attribuer de valeur initiale au champ, et nous ne pouvons pas l'attribuer par champ ou par index. L'affectation provoquera une erreur de panique. 🎜rrreee🎜 Par conséquent, si vous souhaitez accéder aux champs d'une structure dans un autre package dans un package, il doit s'agir d'une variable commençant par une majuscule, c'est-à-dire une variable exportée, telle que : 🎜rrreee🎜Tags🎜🎜Quand En définissant les champs de structure, en plus des noms de champs et des types de données, vous pouvez également utiliser des backticks pour déclarer des méta-informations pour les champs de structure, appelées Tag, qui sont utilisées pour les associer aux champs lors de la phase de compilation. exemple, nous modifions la structure dans l'exemple ci-dessus en :🎜type Member struct { Id int `json:"id,-"` Name string `json:"name"` Email string `json:"email"` Gender int `json:"gender,"` Age int `json:"age"` }
上面例子演示的是使用encoding/json包编码或解码结构体时使用的Tag信息。
Tag由反引号括起来的一系列用空格分隔的key:"value"键值对组成,如:
Id int `json:"id" gorm:"AUTO_INCREMENT"`
下面总结几点结构体的相关特性:
结构体与数组一样,是复合类型,无论是作为实参传递给函数时,还是赋值给其他变量,都是值传递,即复一个副本。
Go语言是支持面向对象编程的,但却没有继承的概念,在结构体中,可以通过组合其他结构体来构建更复杂的结构体。
一个结构体,并没有包含自身,比如Member中的字段不能是Member类型,但却可能是*Member。
在Go语言中,将函数绑定到具体的类型中,则称该函数是该类型的方法,其定义的方式是在func与函数名称之间加上具体类型变量,这个类型变量称为方法接收器
,如:
注意,并不是只有结构体才能绑定方法,任何类型都可以绑定方法,只是我们这里介绍将方法绑定到结构体中。
func setName(m Member,name string){//普通函数 m.Name = name } func (m Member)setName(name string){//绑定到Member结构体的方法 m.Name = name }
从上面的例子中,我们可以看出,通过方法接收器
可以访问结构体的字段,这类似其他编程语言中的this关键词,但在Go语言中,只是一个变量名而已,我们可以任意命名方法接收器
。
调用结构体的方法,与调用字段一样:
m := Member{} m.setName("小明") fmt.Println(m.Name)//输出为空
上面的代码中,我们会很奇怪,不是调用setName()方法设置了字段Name的值了吗?为什么还是输出为空呢?
这是因为,结构体是值传递,当我们调用setName时,方法接收器接收到是只是结构体变量的一个副本,通过副本对值进行修复,并不会影响调用者,因此,我们可以将方法接收器定义为指针变量,就可达到修改结构体的目的了。
func (m *Member)setName(name string){/将Member改为*Member m.Name = name } m := Member{} m.setName("小明") fmt.Println(m.Name)//小明
方法和字段一样,如果首字母为小写,则只允许在包内可见,在其他包中是无法访问的,因此,如果要在其他包中访问setName
,则应该将方法名改为SetName
。
我们知道,结构体中并没有继承的概念,其实,在Go语言中也没有继承的概念,Go语言的编程哲学里,推荐使用组合
的方式来达到代码复用效果。
组合,可以理解为定义一个结构体中,其字段可以是其他的结构体,这样,不同的结构体就可以共用相同的字段。
注意,在记得我们前面提过的,结构体不能包含自身,但可能包含指向自身的结构体指针。
例如,我们定义了一个名为Animal表示动物,如果我们想定义一个结构体表示猫,如:
type Animal struct { Name string //名称 Color string //颜色 Height float32 //身高 Weight float32 //体重 Age int //年龄 } //奔跑 func (a Animal)Run() { fmt.Println(a.Name + "is running") } //吃东西 func (a Animal)Eat() { fmt.Println(a.Name + "is eating") } type Cat struct { a Animal } func main() { var c = Cat{ a: Animal{ Name: "猫猫", Color: "橙色", Weight: 10, Height: 30, Age: 5, }, } fmt.Println(c.a.Name) c.a.Run() }
可以看到,我们定义Cat结构体时,可以把Animal结构体作为Cat的字段。
上面的例子,我们看到,把Animal结构体作为Cat的字段时,其变量名为a,所以我们访问Animal的方法时,语法为c.a.Run()
,这种通过叶子属性访问某个字段类型所带的方法和字段用法非常繁琐。
Go语言支持直接将类型作为结构体的字段,而不需要取变量名,这种字段叫匿名字段
,如:
type Lion struct { Animal //匿名字段 } func main(){ var lion = Lion{ Animal{ Name: "小狮子", Color: "灰色", }, } lion.Run() fmt.Println(lion.Name) }
通过上面例子,可以看到,通过匿名字段组合其他类型,而后访问匿名字段类型所带的方法和字段时,不需要使用叶子属性,非常方便。
在Go语言编程中,结构体大概算是使用得最多的数据类型了,通过定义不同字段和方法的结构体,抽象组合不同的结构体,这大概便是Go语言中对面向对象编程了。
原文地址:https://juejin.cn/post/6844903814168838151
更多编程相关知识,请访问:编程视频!!
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!