Go 语言结构体

Go 语言中数组可以存储同一类型的数据,但在结构体中我们可以为不同项定义不同的数据类型。

结构体是由一系列具有相同类型或不同类型的数据构成的数据集合。

结构体表示一项记录,比如保存图书馆的书籍记录,每本书有以下属性:

  • Title :标题
  • Author : 作者
  • Subject:学科
  • ID:书籍ID

定义结构体

结构体定义需要使用 type 和 struct 语句。struct 语句定义一个新的数据类型,结构体有中有一个或多个成员。type 语句设定了结构体的名称。结构体的格式如下:

type struct_variable_type struct {
member definition;
member definition;
...
member definition;
}

一旦定义了结构体类型,它就能用于变量的声明,语法格式如下:

variable_name := structure_variable_type {value1, value2...valuen}

variable_name := structure_variable_type { key1: value1, key2: value2..., keyn: valuen}
package main

import "fmt"

type Books struct {
title string
author string
subject string
book_id int
} func main() { // 创建一个新的结构体
fmt.Println(Books{"Go 语言", "www.runoob.com", "Go 语言教程", 6495407}) // 也可以使用 key => value 格式
fmt.Println(Books{title: "Go 语言", author: "www.runoob.com", subject: "Go 语言教程", book_id: 6495407}) // 忽略的字段为 0 或 空
fmt.Println(Books{title: "Go 语言", author: "www.runoob.com"})
}

输出结果为:

{Go 语言 www.runoob.com Go 语言教程 6495407}
{Go 语言 www.runoob.com Go 语言教程 6495407}
{Go 语言 www.runoob.com 0}

访问结构体成员

如果要访问结构体成员,需要使用点号 . 操作符,格式为:

结构体.成员名"

结构体类型变量使用 struct 关键字定义,实例如下:

package main

import "fmt"

type Books struct {
title string
author string
subject string
book_id int
} func main() {
var Book1 Books /* 声明 Book1 为 Books 类型 */
var Book2 Books /* 声明 Book2 为 Books 类型 */ /* book 1 描述 */
Book1.title = "Go 语言"
Book1.author = "www.runoob.com"
Book1.subject = "Go 语言教程"
Book1.book_id = 6495407 /* book 2 描述 */
Book2.title = "Python 教程"
Book2.author = "www.runoob.com"
Book2.subject = "Python 语言教程"
Book2.book_id = 6495700 /* 打印 Book1 信息 */
fmt.Printf( "Book 1 title : %s\n", Book1.title)
fmt.Printf( "Book 1 author : %s\n", Book1.author)
fmt.Printf( "Book 1 subject : %s\n", Book1.subject)
fmt.Printf( "Book 1 book_id : %d\n", Book1.book_id) /* 打印 Book2 信息 */
fmt.Printf( "Book 2 title : %s\n", Book2.title)
fmt.Printf( "Book 2 author : %s\n", Book2.author)
fmt.Printf( "Book 2 subject : %s\n", Book2.subject)
fmt.Printf( "Book 2 book_id : %d\n", Book2.book_id)
}

以上实例执行运行结果为:

Book 1 title : Go 语言
Book 1 author : www.runoob.com
Book 1 subject : Go 语言教程
Book 1 book_id : 6495407
Book 2 title : Python 教程
Book 2 author : www.runoob.com
Book 2 subject : Python 语言教程
Book 2 book_id : 6495700

结构体作为函数参数

你可以像其他数据类型一样将结构体类型作为参数传递给函数。并以以上实例的方式访问结构体变量:

package main

import "fmt"

type Books struct {
title string
author string
subject string
book_id int
} func main() {
var Book1 Books /* 声明 Book1 为 Books 类型 */
var Book2 Books /* 声明 Book2 为 Books 类型 */ /* book 1 描述 */
Book1.title = "Go 语言"
Book1.author = "www.runoob.com"
Book1.subject = "Go 语言教程"
Book1.book_id = 6495407 /* book 2 描述 */
Book2.title = "Python 教程"
Book2.author = "www.runoob.com"
Book2.subject = "Python 语言教程"
Book2.book_id = 6495700 /* 打印 Book1 信息 */
printBook(Book1) /* 打印 Book2 信息 */
printBook(Book2)
} func printBook( book Books ) {
fmt.Printf( "Book title : %s\n", book.title);
fmt.Printf( "Book author : %s\n", book.author);
fmt.Printf( "Book subject : %s\n", book.subject);
fmt.Printf( "Book book_id : %d\n", book.book_id);
}

以上实例执行运行结果为:

Book title : Go 语言
Book author : www.runoob.com
Book subject : Go 语言教程
Book book_id : 6495407
Book title : Python 教程
Book author : www.runoob.com
Book subject : Python 语言教程
Book book_id : 6495700

结构体指针

你可以定义指向结构体的指针类似于其他指针变量,格式如下:

var struct_pointer *Books

以上定义的指针变量可以存储结构体变量的地址。查看结构体变量地址,可以将 & 符号放置于结构体变量前:

struct_pointer = &Book1;
使用结构体指针访问结构体成员,使用 "." 操作符:
struct_pointer.title;

接下来让我们使用结构体指针重写以上实例,代码如下

package main

import "fmt"

type Books struct {
title string
author string
subject string
book_id int
} func main() {
var Book1 Books /* Declare Book1 of type Book */
var Book2 Books /* Declare Book2 of type Book */ /* book 1 描述 */
Book1.title = "Go 语言"
Book1.author = "www.runoob.com"
Book1.subject = "Go 语言教程"
Book1.book_id = 6495407 /* book 2 描述 */
Book2.title = "Python 教程"
Book2.author = "www.runoob.com"
Book2.subject = "Python 语言教程"
Book2.book_id = 6495700 /* 打印 Book1 信息 */
printBook(&Book1) /* 打印 Book2 信息 */
printBook(&Book2)
}
func printBook( book *Books ) {
fmt.Printf( "Book title : %s\n", book.title);
fmt.Printf( "Book author : %s\n", book.author);
fmt.Printf( "Book subject : %s\n", book.subject);
fmt.Printf( "Book book_id : %d\n", book.book_id);
}

以上实例执行运行结果为:

Book title : Go 语言
Book author : www.runoob.com
Book subject : Go 语言教程
Book book_id : 6495407
Book title : Python 教程
Book author : www.runoob.com
Book subject : Python 语言教程
Book book_id : 6495700
struct 类似于 java 中的类,可以在 struct 中定义成员变量。

要访问成员变量,可以有两种方式:

    1.通过 struct 变量.成员 变量来访问。
2.通过s truct 指针.成员 变量来访问。 不需要通过 getter, setter 来设置访问权限。 type Rect struct{ //定义矩形类
x,y float64 //类型只包含属性,并没有方法
width,height float64
}
func (r *Rect) Area() float64{ //为Rect类型绑定Area的方法,*Rect为指针引用可以修改传入参数的值
return r.width*r.height //方法归属于类型,不归属于具体的对象,声明该类型的对象即可调用该类型的方法
}

1、go语言的结构体

package main

import "fmt"

//go语言没有类,只有结构体,struct,go语言的结构体就相当于python中的类

//定义一个结构体
//type 结构体名 struct {
//
//} type Student struct {
id int
name string
sex byte
age int
addr string } func main() {
//1、顺序实例化一个结构体
var s1 Student = Student{} fmt.Println(s1)
//{1 test1 102 18 beijing} s2 := Student{id:2,name:"test2",sex:'m',age:18,addr:"beijing"} fmt.Println(s2)
//{2 test2 109 18 beijing} //指定初始化成员,如果有的字段没有赋值,则用默认值代替
s3 := Student{id:2,name:"test2",age:18}
fmt.Println(s3)
//{2 test2 0 18 } //取结构体中的值,用点来取
fmt.Println(s3.age)
//18 //结构体作为指针变量初始化 var s4 *Student = &Student{id:1,name:"test1",sex:'f',age:18,addr:"beijing"}
fmt.Println(s4)
//&{1 test1 102 18 beijing} //下面两种都可以取指针对象的属性
fmt.Println((*s4).name)
//test1
fmt.Println(s4.name)
//test1 //定义一个指针变量的结构体
s5 := &Student{id:1,name:"test1",sex:'f',age:18,addr:"beijing"}
fmt.Println(s5)
//&{1 test1 102 18 beijing} }

  

这里我们可以看到,如果传递一个指针进去,可以使用“(&指针name).”的方式调用方法,也可以直接使用"指针name."的方法调用方法,这个是go在内部替我们做了封装

2、结构体作为参数传递给方法

package main

import "fmt"

type Student struct {
id int
name string
sex byte
age int
addr string } //结构体作为参数传递给方法
//定义一个传递学生对象的方法
func tmpStuent(tmp Student) {
tmp.id = 250
fmt.Println(tmp)
} //定义一个传递指针对象的方法
func tmpStudentobj(p *Student) {
p.id = 249
fmt.Println(p)
} func main() {
var s27_1 Student = Student{id:1,name:"test1",sex:'f',age:18,addr:"beijing"}
tmpStuent(s27_1)
//{250 test1 102 18 beijing} fmt.Println(s27_1)
//{1 test1 102 18 beijing} tmpStudentobj(&s27_1)
//&{249 test1 102 18 beijing}
fmt.Println(s27_1)
//{249 test1 102 18 beijing} }

  

通过上面的例子我们知道

如果传递一个结构体给函数,在函数中修改这个结构体,不会影响外面的结构体的值,因为他是重新复制了一份数据

但是,如果传递一个结构体的指针给函数,在函数中修改这个指针的值,是会影响外面结构体的值的,因为他们是公用一份数据

3、匿名字段,通过匿名字段来实现继承父类的字段

package main

import "fmt"

//go语言的面向对象非常的优雅和简单
//没有封装,多态,继承这些概念,但是同样通过其他方式来实现上面的功能 //封装:通过方法来实现
//继承:通过匿名字段来实现
//多态:通过接口来实现 type Person struct {
name string
sex byte
age int } type Student1 struct {
//匿名字段 //默认Student_1这个结构体包含了Person这个结构体的所有的字段
Person
//还可以定义自己私有的字段
i int
addr string } type Student2 struct {
//匿名字段 //默认Student_1这个结构体包含了Person这个结构体的所有的字段
Person
//还可以定义自己私有的字段
i int
addr string
name string } type myint int type Student3 struct {
//自定义类型
Person //内置类型
int //自定义类型
myint
} //指针类型的结构体
type Student4 struct {
//匿名字段 //默认Student_1这个结构体包含了Person这个结构体的所有的字段
*Person
//还可以定义自己私有的字段
i int
addr string
name string } func main() {
//使用匿名字段初始化一个子类
s28_1 := Student1{Person:Person{name:"test1",sex:'m',age:18},addr:"beijing",i:1}
fmt.Println(s28_1) //打印的结构是一个结构体包含一个结构体
//{{test1 109 18} 1 beijing} //这里如果不这样写就会报错Person:Person
//mixture of field:value and value initializers //同名字段的情况 //如果有同名字段,则默认是就近原则
var test28_2 Student2
//test28_2.name = "test"
//fmt.Println(test28_2)
//{{ 0 0} 0 test} //如果要想给父类的name赋值怎么做 test28_2.Person.name = "abc"
fmt.Println(test28_2)
//{{abc 0 0} 0 } //所有的内置类型和自定义类型都可以作为匿名字段使用
//我们这里的myint和Python都可以当做一个自定义类型使用 test28_3 := Student3{Person:Person{name:"test1",sex:'m',age:18},int:1,myint:2}
fmt.Println(test28_3)
//{{test1 109 18} 1 2} fmt.Println(test28_3.myint)
//2 //指针类型的匿名字段
s28_4 := Student4{Person:&Person{name:"test1",sex:'m',age:18},addr:"beijing",i:1} fmt.Println(s28_4)
//{0x1f43e0e0 1 beijing } fmt.Println(s28_4.age,"11")
//18 11
}

  

  

匿名字段实现继承

package main

import "fmt"

type Student struct {
sid int
name string
age int
sex byte
addr string } type Student1 struct {
Student hobby string
addr int }
var s1 Student = Student{
sid:1,
name:"test",
age:12,
sex:'m',
addr:"sz",
} func test10_1(s Student) {
s.name = "test10_1"
fmt.Println(s)
} func test10_2(s *Student) {
s.name = "test10_2"
fmt.Println(s)
} func main() {
fmt.Println(s1)
fmt.Println(s1.name) //这种方式必须在函数中使用
s2 := Student{sid:1,name:"test2",age:13,sex:'f',addr:"nmg"}
fmt.Println(s2) s3 := Student{age:13,sex:'f',addr:"nmg"} //如果在有未传的变量,则go会给一个默认值
fmt.Println(s3.name,s3.sid,s3.addr) //定义一个指针类型的结构体变量
var s4 *Student = &Student{sid:2,name:"test4",age:14,sex:'f',addr:"nmg1"} //通过下面两种方式都可以访问指针变量指向的结构体变量的值
fmt.Println(s4.addr)
fmt.Println((*s4).addr) s5 := &Student{sid:2,name:"test4",age:14,sex:'f',addr:"nmg1"}
fmt.Println(s5) //test10_1(s2)
//fmt.Println(s2.name) test10_2(s5)
fmt.Println(s5) s6 := Student1{Student:Student{sid:2,name:"test4",age:14,sex:'f',addr:"nmg1"},hobby:"football",addr:12} //通过匿名字段实现继承,默认情况下可以访问父类和自己的字段,但是如果父类的字段和自己的字段是相同的,则默认会访问自己的字段
//如果一定要访问父类的字段,则需要这样访问s6.Student.addr
fmt.Println(s6.sid,s6.hobby,s6.addr,s6.Student.addr) }

go语言之面向对象的更多相关文章

  1. [java学习笔记]java语言核心----面向对象之this关键字

    一.this关键字 体现:当成员变量和函数的局部变量重名时,可以使用this关键字来区别:在构造函数中调用其它构造函数 原理:         代表的是当前对象.         this就是所在函数 ...

  2. 第二十五节:Java语言基础-面向对象基础

    面向对象 面向过程的代表主要是C语言,面向对象是相对面向过程而言,Java是面向对象的编程语言,面向过程是通过函数体现,面向过程主要是功能行为. 而对于面向对象而言,将功能封装到对象,所以面向对象是基 ...

  3. 如何使用C语言的面向对象

    我们都知道,C++才是面向对象的语言,但是C语言是否能使用面向对象的功能? (1)继承性 typedef struct _parent { int data_parent; }Parent; type ...

  4. 比较分析C++、Java、Python、R语言的面向对象特征,这些特征如何实现的?有什么相同点?

    一门课的课后题答案,在这里备份一下: 面向对象程序设计语言 –  比较分析C++.Java.Python.R语言的面向对象特征,这些特征如何实现的?有什么相同点? C++ 语言的面向对象特征: 对象模 ...

  5. 用C语言实现面向对象的开发

    C语言的对象化模型 面向对象的特征主要包括: .封装,隐藏内部实现 .继承,复用现有代码 .多态,改写对象行为 采用C语言实现的关键是如何运用C语言本身的特性来实现上述面向对象的特征. 1.1 封装 ...

  6. 基于C语言的面向对象编程

    嵌入式软件开发中,虽然很多的开发工具已经支持C++的开发,但是因为有时考虑运行效率和编程习惯,还是有很多人喜欢用C来开发嵌入式软件.Miro Samek说:"我在开发现场发现,很多嵌入式软件 ...

  7. 已看1.熟练的使用Java语言进行面向对象程序设计,有良好的编程习惯,熟悉常用的Java API,包括集合框架、多线程(并发编程)、I/O(NIO)、Socket、JDBC、XML、反射等。[泛型]\

    1.熟练的使用Java语言进行面向对象程序设计,有良好的编程习惯,熟悉常用的Java API,包括集合框架.多线程(并发编程).I/O(NIO).Socket.JDBC.XML.反射等.[泛型]\1* ...

  8. luajit利用ffi结合C语言实现面向对象的封装库

    luajit中.利用ffi能够嵌入C.眼下luajit的最新版是2.0.4,在这之前的版本号我还不清楚这个扩展库详细怎么样,只是在2.04中,真的非常爽.  既然是嵌入C代码.那么要说让lua支持 ...

  9. C 语言实现面向对象编程

    转载 https://blog.csdn.net/onlyshi/article/details/81672279 C 语言实现面向对象编程1.引言面向对象编程(OOP)并不是一种特定的语言或者工具, ...

随机推荐

  1. 如果你不了解Java的JVM,那真的很难进BAT一线大厂!

    前言 对于开发人员来说,如果不了解Java的JVM,那真的是很难写得一手好代码,很难查得一手好bug.同时,JVM也是面试环节的中重灾区.我们不能为了面试而面试,但是学习会这些核心知识你必定会成为面试 ...

  2. SpringBoot集成Swagger2在线文档

    目录 SpringBoot集成Swagger2在线文档 前言 集成SpringBoot 登录接口文档示例 代码 效果 注解说明 总结 SpringBoot集成Swagger2在线文档 前言 不得不说, ...

  3. Maven使用教程三:maven的生命周期及插件机制详解

    前言 今天这个算是学习Maven的一个收尾文章,里面内容不局限于标题中提到的,后面还加上了公司实际使用的根据profile配置项目环境以及公司现在用的archetype 模板等例子. 后面还会总结一个 ...

  4. bsoj5988 [Achen模拟赛]期望 题解

    bsoj5988 Description [题目背景] NOI2018 已经过去了许久,2019 届的 BSOIer 们退役的退役,颓废的颓废,计数能力大不如前.曾经的数数之王 xxyj 坦言:&qu ...

  5. 如何优雅地关闭worker进程?

    之前我们讲解 Nginx 命令行的时候,可以看到 Nginx 停止有两种方式,分别是 nginx -s quit 和 nginx -s stop,其中 stop 是指立即停止 Nginx,而 quit ...

  6. 重新精读《Java 编程思想》系列之类的访问权限

    Java 中,我们用访问权限修饰词确定库中的哪些类对于使用者是可以使用的. 访问权限修饰词有 public,protected,private 和什么都不写. 那么对于类来说,我们只可以用 publi ...

  7. 探索JAVA并发 - 可重入锁和不可重入锁

    本人免费整理了Java高级资料,涵盖了Java.Redis.MongoDB.MySQL.Zookeeper.Spring Cloud.Dubbo高并发分布式等教程,一共30G,需要自己领取.传送门:h ...

  8. 错题本:ConstraintLayout 不能正常显示

    理想效果: 实际效果: 原因:因为文件中一个控件的约束属性写错了 这个属性是 app:layout_constraintLeft_toLeftOf="@id/oa_setting_group ...

  9. Mysql相关知识总结-持续更新~~~

    2019-12-11对varchar类型排序问题的解决 在mysql默认order by 只对数字与日期类型可以排序,但对于varchar字符型类型排序好像没有用了,下面我来给各位同学介绍varcha ...

  10. Java_垃圾回收算法

    参考:<深入理解JAVA虚拟机>第二版 3.3 垃圾收集算法 由于垃圾收集算法的实现涉及大量的程序细节,而且各个平台的虚拟机操作内存的方法又各不相同,只是介绍几种算法的思想及其发展过程. ...