GO语言结构体面向对象操作示例

目录
  • 匿名字段初始化
  • 成员的操作
  • 同名字段
  • 非结构体匿名字段
  • 结构体指针类型匿名字段
  • 面向过程和对象函数的区别
  • 为结构体类型添加方法
  • 指针变量方法集
  • 普通变量方法集
  • 方法的继承
  • 方法的重写
  • 方法值
  • 方法表达式

匿名字段初始化

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别
	age  int    //年龄
}
type Student struct {
	Person //只有类型,没有名字,匿名字段,继承了Person的成员
	id     int
	addr   string
}
func main() {
	//顺序初始化
	var s1 Student = Student{Person{"mike", 'm', 18}, 1, "bj"}
	fmt.Println("s1 = ", s1)
	//自动推导类型
	s2 := Student{Person{"mike", 'm', 18}, 1, "bj"}
	//fmt.Println("s2 = ", s2)
	//%+v, 显示更详细
	fmt.Printf("s2 = %+v\n", s2)
	//指定成员初始化,没有初始化的常用自动赋值为0
	s3 := Student{id: 1}
	fmt.Printf("s3 = %+v\n", s3)
	s4 := Student{Person: Person{name: "mike"}, id: 1}
	fmt.Printf("s4 = %+v\n", s4)
	//s5 := Student{"mike", 'm', 18, 1, "bj"} //err
}

成员的操作

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
type Student struct {
	Person //只有类型,没有名字,匿名字段,继承了Person的成员
	id     int
	addr   string
}
func main() {
	s1 := Student{Person{"mike", 'm', 18}, 1, "bj"}
	s1.name = "yoyo"
	s1.sex = 'f'
	s1.age = 22
	s1.id = 666
	s1.addr = "sz"
	s1.Person = Person{"go", 'm', 18}
	fmt.Println(s1.name, s1.sex, s1.age, s1.id, s1.addr)
}

同名字段

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
type Student struct {
	Person //只有类型,没有名字,匿名字段,继承了Person的成员
	id     int
	addr   string
	name   string //和Person同名了
}
func main() {
	//声明(定义一个变量)
	var s Student
	//默认规则(纠结原则),如果能在本作用域找到此成员,就操作此成员
	//					如果没有找到,找到继承的字段
	s.name = "mike" //操作的是Student的name,还是Person的name?, 结论为Student的
	s.sex = 'm'
	s.age = 18
	s.addr = "bj"
	//显式调用
	s.Person.name = "yoyo" //Person的name
	fmt.Printf("s = %+v\n", s)
}

非结构体匿名字段

package main
import "fmt"
type mystr string //自定义类型,给一个类型改名
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
type Student struct {
	Person //结构体匿名字段
	int    //基础类型的匿名字段
	mystr
}
func main() {
	s := Student{Person{"mike", 'm', 18}, 666, "hehehe"}
	fmt.Printf("s = %+v\n", s) //s = {Person:{name:mike sex:109 age:18} int:666 mystr:hehehe}
	s.Person = Person{"go", 'm', 22}
	fmt.Println(s.name, s.age, s.sex, s.int, s.mystr)  // go 22 109 666 hehehe
	fmt.Println(s.Person, s.int, s.mystr)  // {go 109 22} 666 hehehe
}

结构体指针类型匿名字段

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
type Student struct {
	*Person //指针类型
	id      int
	addr    string
}
func main() {
	s1 := Student{&Person{"mike", 'm', 18}, 666, "bj"}
	fmt.Println(s1.name, s1.sex, s1.age, s1.id, s1.addr)
	//先定义变量
	var s2 Student
	s2.Person = new(Person) //分配空间
	s2.name = "yoyo"
	s2.sex = 'm'
	s2.age = 18
	s2.id = 222
	s2.addr = "sz"
	fmt.Println(s2.name, s2.sex, s2.age, s2.id, s2.addr)
}

面向过程和对象函数的区别

package main
import "fmt"
//实现2数相加
//面向过程
func Add01(a, b int) int {
	return a + b
}
//面向对象,方法:给某个类型绑定一个函数
type long int
//tmp叫接收者,接收者就是传递的一个参数
func (tmp long) Add02(other long) long {
	return tmp + other
}
func main() {
	//var result int
	//result = Add01(1, 1) //普通函数调用方式
	//fmt.Println("result = ", result)
	//定义一个变量
	var a long = 2
	//调用方法格式: 变量名.函数(所需参数)
	r := a.Add02(3)
	fmt.Println("r = ", r)
	//面向对象只是换了一种表现形式
}

为结构体类型添加方法

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
//带有接收者的函数叫方法
func (tmp Person) PrintInfo() {
	fmt.Println("tmp = ", tmp)
}
//通过一个函数,给成员赋值
func (p *Person) SetInfo(n string, s byte, a int) {
	p.name = n
	p.sex = s
	p.age = a
}
func main() {
	//定义同时初始化
	p := Person{"mike", 'm', 18}
	p.PrintInfo()  //tmp =  {mike 109 18}
	//定义一个结构体变量
	var p2 Person
	(&p2).SetInfo("yoyo", 'f', 22)
	p2.PrintInfo() //tmp =  {yoyo 102 22}
}

指针变量方法集

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
func (p Person) SetInfoValue() {
	fmt.Println("SetInfoValue")
}
func (p *Person) SetInfoPointer() {
	fmt.Println("SetInfoPointer")
}
func main() {
	//结构体变量是一个指针变量,它能够调用哪些方法,这些方法就是一个集合,简称方法集
	p := &Person{"mike", 'm', 18}
	p.SetInfoPointer() //func (p *Person) SetInfoPointer()
	(*p).SetInfoPointer() //把(*p)转换层p后再调用,等价于上面
	//内部做的转换, 先把指针p, 转成*p后再调用
	//(*p).SetInfoValue()  //SetInfoValue
	//p.SetInfoValue()  // SetInfoValue
}

普通变量方法集

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
func (p Person) SetInfoValue() {
	fmt.Println("SetInfoValue")
}
func (p *Person) SetInfoPointer() {
	fmt.Println("SetInfoPointer")
}
func main() {
	p := Person{"mike", 'm', 18}
	p.SetInfoPointer() //func (p *Person) SetInfoPointer()
	//内部,先把p, 转为为&p再调用, (&p).SetInfoPointer()
	p.SetInfoValue()  //直接调用,内部不需要转
}

方法的继承

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
//Person类型,实现了一个方法
func (tmp *Person) PrintInfo() {
	fmt.Printf("name=%s, sex=%c, age=%d\n", tmp.name, tmp.sex, tmp.age)
}
//有个学生,继承Person字段,成员和方法都继承了
type Student struct {
	Person //匿名字段
	id     int
	addr   string
}
func main() {
	s := Student{Person{"mike", 'm', 18}, 666, "bj"}
	s.PrintInfo()
}

方法的重写

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
//Person类型,实现了一个方法
func (tmp *Person) PrintInfo() {
	fmt.Printf("name=%s, sex=%c, age=%d\n", tmp.name, tmp.sex, tmp.age)
}
//有个学生,继承Person字段,成员和方法都继承了
type Student struct {
	Person //匿名字段
	id     int
	addr   string
}
//Student也实现了一个方法,这个方法和Person方法同名,这种方法叫重写
func (tmp *Student) PrintInfo() {
	fmt.Println("Student: tmp = ", tmp)
}
func main() {
	s := Student{Person{"mike", 'm', 18}, 666, "bj"}
	//就近原则:先找本作用域的方法,找不到再用继承的方法
	s.PrintInfo() //到底调用的是Person, 还是Student, 结论是Student
	//显式调用继承的方法
	s.Person.PrintInfo()
}

方法值

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
func (p Person) SetInfoValue() {
	fmt.Printf("SetInfoValue: %p, %v\n", &p, p)
}
func (p *Person) SetInfoPointer() {
	fmt.Printf("SetInfoPointer: %p, %v\n", p, p)
}
func main() {
	p := Person{"mike", 'm', 18}
	fmt.Printf("main: %p, %v\n", &p, p)
	//p.SetInfoPointer() //传统调用方式
	//保存方式入口地址
	pFunc := p.SetInfoPointer //这个就是方法值,调用函数时,无需再传递接收者,隐藏了接收者
	pFunc()                   //等价于 p.SetInfoPointer()
	vFunc := p.SetInfoValue
	vFunc() //等价于 p.SetInfoValue()
}

方法表达式

package main
import "fmt"
type Person struct {
	name string //名字
	sex  byte   //性别, 字符类型
	age  int    //年龄
}
func (p Person) SetInfoValue() {
	fmt.Printf("SetInfoValue: %p, %v\n", &p, p)
}
func (p *Person) SetInfoPointer() {
	fmt.Printf("SetInfoPointer: %p, %v\n", p, p)
}
func main() {
	p := Person{"mike", 'm', 18}
	fmt.Printf("main: %p, %v\n", &p, p)
	//方法值   f := p.SetInfoPointer //隐藏了接收者
	//方法表达式
	f := (*Person).SetInfoPointer
	f(&p) //显式把接收者传递过去 ====》 p.SetInfoPointer()
	f2 := (Person).SetInfoValue
	f2(p) //显式把接收者传递过去 ====》 p.SetInfoValue()
}

以上就是GO语言结构体面向对象操作示例的详细内容,更多关于GO语言结构体面向对象的资料请关注我们其它相关文章!

(0)

相关推荐

  • 深入理解GO语言的面向对象

    前言 有过C++语言学习经历的朋友都知道,面向对象主要包括了三个基本特征:封装.继承和多态.封装,就是指运行的数据和函数绑定在一起,C++中主要是通过this指针来完成的:继承,就是指class之间可以相互继承属性和函数:多态,主要就是用统一的接口来处理通用的逻辑,每个class只需要按照接口实现自己的回调函数就可以了. 作为集大成者的Go语言,自然不会在面向对象上面无所作为.相比较C++.Java.C#等面向对象语言而言,它的面向对象更简单,也更容易理解. go语言中并没有像C++,Java语

  • golang修改结构体中的切片值方法

    golang修改结构体中的切片值,直接传结构体地址就可以 package main import "fmt" type rspInfo struct { KeyWords string `json:"key_words"` Value []string `json:"value"` } func setSlice(te *[]string){ str := "12" *te = append(*te,str) } //结构提传

  • django框架面向对象ORM模型继承用法实例分析

    本文实例讲述了django框架面向对象ORM模型继承用法.分享给大家供大家参考,具体如下: Django ORM对模型继承的支持,将python面向对象的编程方法与数据库面向关系表的数据结构结合的很巧妙.支持三种风格的模型继承. 1.抽象类继承:父类继承自models.Model,但不会在底层数据库中生成相应的数据表,父类的属性列存储在其子类的数据表中. 2.多表继承:多表继承的每个模型类都在底层数据库中生成相应的数据表管理数据. 3.代理模型继承:父类用于在底层数据库中管理数据表,而子类不定义

  • Golang自定义结构体转map的操作

    在Golang中,如何将一个结构体转成map? 本文介绍两种方法.第一种是是使用json包解析解码编码.第二种是使用反射,使用反射的效率比较高,代码在这里.如果觉得代码有用,可以给我的代码仓库一个star. 假设有下面的一个结构体 func newUser() User { name := "user" MyGithub := GithubPage{ URL: "https://github.com/liangyaopei", Star: 1, } NoDive :

  • golang 实现两个结构体复制字段

    实际工作中可能会有这样的场景: 两个结构体(可能类型一样), 字段名和类型都一样, 想复制一个结构体的全部或者其中某几个字段的值到另一个(即merge操作), 自然想到可以用反射实现 package main import "fmt" import "reflect" // 用b的所有字段覆盖a的 // 如果fields不为空, 表示用b的特定字段覆盖a的 // a应该为结构体指针 func CopyFields(a interface{}, b interface

  • GO语言结构体面向对象操作示例

    目录 匿名字段初始化 成员的操作 同名字段 非结构体匿名字段 结构体指针类型匿名字段 面向过程和对象函数的区别 为结构体类型添加方法 指针变量方法集 普通变量方法集 方法的继承 方法的重写 方法值 方法表达式 匿名字段初始化 package main import "fmt" type Person struct { name string //名字 sex byte //性别 age int //年龄 } type Student struct { Person //只有类型,没有名

  • go语言结构体指针操作示例详解

    目录 指针 go指针操作 不能操作不合法指向 new函数 指针做函数的参数 数组指针 结构体指针变量 结构体成员普通变量 结构体成员指针变量 结构体比较和赋值 结构体作为函数参数 指针 指针是代表某个内存地址的值.内存地址储存另一个变量的值. 指针(地址),一旦定义了不可改变,指针指向的值可以改变 go指针操作 1.默认值nil,没有NULL常量 2.操作符“&”取变量地址,“*“通过指针(地址)访问目标对象(指向值) 3.不支持指针运算,不支持“->”(箭头)运算符,直接用“.”访问目标成

  • C语言 结构体数组详解及示例代码

    所谓结构体数组,是指数组中的每个元素都是一个结构体.在实际应用中,结构体数组常被用来表示一个拥有相同数据结构的群体,比如一个班的学生.一个车间的职工等. 定义结构体数组和定义结构体变量的方式类似,请看下面的例子: struct stu{ char *name; //姓名 int num; //学号 int age; //年龄 char group; //所在小组 float score; //成绩 }class[5]; 表示一个班级有5个学生. 结构体数组在定义的同时也可以初始化,例如: str

  • go语言通过结构体生成json示例解析

    目录 通过结构体生成json 通过map生成json json解析到结构体 json解析到map 通过结构体生成json buf, err := json.MarshalIndent(s, "", " ") //格式化编码 package main import ( "encoding/json" "fmt" ) //成员变量名首字母必须大写 type IT struct { Company string `json:&quo

  • C语言 结构体(Struct)详解及示例代码

    前面的教程中我们讲解了数组(Array),它是一组具有相同类型的数据的集合.但在实际的编程过程中,我们往往还需要一组类型不同的数据,例如对于学生信息登记表,姓名为字符串,学号为整数,年龄为整数,所在的学习小组为字符,成绩为小数,因为数据类型不同,显然不能用一个数组来存放. 在C语言中,可以使用结构体(Struct)来存放一组不同类型的数据.结构体的定义形式为: struct 结构体名{     结构体所包含的变量或数组 }; 结构体是一种集合,它里面包含了多个变量或数组,它们的类型可以相同,也可

  • C语言结构体(struct)常见使用方法(细节问题)

    基本定义:结构体,通俗讲就像是打包封装,把一些有共同特征(比如同属于某一类事物的属性,往往是某种业务相关属性的聚合)的变量封装在内部,通过一定方法访问修改内部变量. 结构体定义: 第一种:只有结构体定义 struct stuff{ char job[20]; int age; float height; }; 第二种:附加该结构体类型的"结构体变量"的初始化的结构体定义 //直接带变量名Huqinwei struct stuff{ char job[20]; int age; floa

  • php读取二进制流(C语言结构体struct数据文件)的深入解析

    尽管php是用C语言开发的,不过令我不解的是php没有提供对结构体struct的直接支持.不过php提供了pack和unpack函数,用来进行二进制数据(binary data)和php内部数据的互转: 复制代码 代码如下: string pack ( string $format [, mixed $args [, mixed $...]] )   //Pack given arguments into binary string according to format.  array unp

  • 利用C语言结构体实现通讯录

    本文实例为大家分享了C语言结构体实现通讯录的具体代码,供大家参考,具体内容如下 用来存储1000个人的信息的通讯录,每个人的信息包括: 姓名.性别.年龄.电话.住址 程序如下: #include<stdio.h> #include<string.h> #include<stdlib.h> struct People { char name[20]; char sex[5]; int age; char tel[15]; char addr[50]; }; //定义人的信

  • C语言结构体的一些理解

    前言 最忙碌的一周已然结束,疲惫之余想和大家聊一聊对C语言中结构体的认识.水平不足,如有问题请大家指正. 一.对结构体的理解 我们知道C语言中有基本数据类型: 1.整(数)型: int:基本整型,用于存储整数,占4个字节,默认值为0 short:短整型,占2个字节 long:长整型,占4个字节 long long:双长整型,占8个字节,比较少用 2.浮点型 float:单精度浮点型,占4个字节 double:双精度浮点型,占8个字节 3.字符型 char:字符型,用于存储单字符,占1个字节. (

  • C语言结构体计算内存占用问题解析

        c语言中结构体使用是非常广泛的,但是结构体有一个问题,就是如果开头的字段属性是字符类型(char),紧跟着的是其他类型,比如整型.长整型.双精度.浮点型,这时候结构体的大小会发生改变,下面给出一个示例: #include <stdio.h> struct person{ char sex; int age; char name[8]; }; int main() { printf("sizeof(person) = %d\n",sizeof(struct perso

随机推荐