您的位置:首页 > 教程 > 服务端类 > Golang > Go语言学习之结构体和方法使用详解

Go语言学习之结构体和方法使用详解

2022-04-21 12:14:45 来源:易采站长站 作者:

Go语言学习之结构体和方法使用详解

目录
1. 结构体别名定义2. 工厂模式3. Tag 原信息4. 匿名字段5. 方法

1. 结构体别名定义

变量别名定义

package main

import "fmt"

type integer int

func main() {
	//类型别名定义
	var i integer = 1000
	fmt.Printf("值: %d, 类型: %T\n", i, i)

	var j int = 100
	j = int(i) 			//j和i不属于同一类型,需要转换
	fmt.Println(j)
}

输出结果如下

值: 1000, 类型: main.integer
1000

结构体别名定义

package main

import "fmt"

//创建结构体Student
type Student struct {
	Number int
}

//结构体定义别名
type Stu Student

func main() {
	//声明Student类型结构体
	var a Student
	a = Student{30}

	//声明Stu类型结构体
	var b Stu
	b = Stu{20}

	//强转类型后才能进行赋值
	a = Student(b)
	fmt.Printf("a = %d,类型: %T\n", a, a)
	b = Stu(a)
	fmt.Printf("b = %d,类型: %T\n", b, b)
}

输出结果如下

a = {20},类型: main.Student
b = {20},类型: main.Stu

2. 工厂模式

Go 中所谓的工厂模式其实就是:

包内一个不可直接实例的结构体(结构体名称首字母小写),包外不可直接实例,那么为了解决这个问题,就写一个包外可调用的函数,通过这个函数实现返回结构体对象。

package main

import "fmt"

type Student struct {
	Name string
	Age  int
}

func main() {
	//初始化
	stu1 := new(Student)
	fmt.Println(stu1)

	//工厂模式处理
	stu2 := NewStudent("张三", 18)
	fmt.Println(stu2)
}

//工厂模式
func NewStudent(name string, age int) *Student {
	return &Student{
		Name: name,
		Age:  age,
	}
}

输出结果如下

&{ 0}
&{张三 18}

总结:① make 用来创建mapslicechannel

② new 用来创建值类型

3. Tag 原信息

在和其他语言进行对接交互时使用JSON格式,有些语言格式大小写规范比较严格,为了使Go语言和其他语言对接数据传输,所以使用Tag原信息进行解决

通俗的来说就相当于是一个充电的转接口

示例

package main

import (
	"encoding/json"
	"fmt"
)

type Student struct {
	Name  string
	Age   int
	Score float32
}

func main() {
	//初始化
	var stu = new(Student)
	stu.Name = "stu"
	stu.Age = 20
	stu.Score = 88

	//使用Json处理结构体,转换成字节数组
	data, err := json.Marshal(stu)
	if err != nil {
		fmt.Println("错误提示:", err)
		return
	}
	fmt.Println(data)				//字节数组形式输出
	fmt.Println(string(data))		//转换成字符串输出
}

输出结果如下

[123 34 78 97 109 101 34 58 34 115 116 117 34 44 34 65 103 101 34 58 50 48 44 34 83 99 111 114 101 34 58 56 56 125]
{"Name":"stu","Age":20,"Score":88}

JSON格式化字段名

package main

import (
	"encoding/json"
	"fmt"
)

type Student struct {
	//json打包时字段名
	Name  string  `json:"stu_name"`
	Age   int     `json:"stu_age"`
	Score float32 `json:"stu_score"`
}

func main() {
	//初始化
	var stu = new(Student)
	stu.Name = "stu"
	stu.Age = 20
	stu.Score = 88

	//使用Json处理结构体,转换成字节数组
	data, err := json.Marshal(stu)
	if err != nil {
		fmt.Println("错误提示:", err)
		return
	}
	fmt.Println(data)
	fmt.Println(string(data))
}

输出结果如下

[123 34 115 116 117 95 110 97 109 101 34 58 34 115 116 117 34 44 34 115 116 117 95 97 103 101 34 58 50 48 44 34 115 116 117 95 115 99 111 114 101 34 58 56 56 125]
{"stu_name":"stu","stu_age":20,"stu_score":88}

4. 匿名字段

结构体中的字段(属性)没有名称,称之为匿名字段

示例

package main

import "fmt"

type Cart struct {
	name  string
	color string
}

type Train struct {
	//匿名字段
	Cart //实现继承
	int  //数据类型定义,仅能存在一次,两个int则会冲突
}

func main() {
	//初始化赋值
	var t Train		
	t.name = "train"
	t.color = "red"
	t.int = 10		//直接调用数据类型赋值
	fmt.Println(t)
}

输出结果如下

{{train red} 10}

双引用结构体,多继承(继承的两个结构体中定义相同属性)

package main

import "fmt"

//父结构体
type Cart struct {
	name  string
	color string
}

//父结构体
type Box struct {
	color string
}

//子结构体
type Train struct {
	//匿名字段
	Cart //实现继承
	Box
	int //数据类型定义,仅能存在一次,两个int则会冲突
}

func main() {
	//初始化赋值
	var t Train
	t.name = "train"
	t.Cart.color = "red"
	t.Box.color = "blue"
	t.int = 10 //直接调用数据类型赋值
	fmt.Println(t)
}

输出结果如下

{{train red} {blue} 10}

package main

import "fmt"

//父结构体
type Cart struct {
	name  string
	color string
}

//父结构体
type Box struct {
	color string
}

//子结构体
type Train struct {
	//匿名字段
	Cart //实现继承
	Box
	int   //数据类型定义,仅能存在一次,两个int则会冲突
	color string
}

func main() {
	//初始化赋值
	var t Train
	t.name = "train"
	t.Cart.color = "red" //Cart的属性
	t.Box.color = "blue" //Box的属性
	t.color = "yellow"   //train自身属性
	t.int = 10           //直接调用数据类型赋值
	fmt.Println(t)
}

5. 方法

Go 中的方法是作用在特定类型的变量上,因此自定义类型,都可以有方法,而不仅仅是 struct语法格式如下
func (recevier type) methodName(参数列表)(返回值){}

​​​​​​​recevier type     特定类型,如指针、别名,结构体 
methodName           方法名

示例

package main

import "fmt"

//定义结构体
type Student struct {
	Name string
	Age  int
}

//定义方法
func (s Student) init(name string, age int) Student {
	s.Name = name
	s.Age = age
	return s
}

func main() {
	var stu Student
	s := stu.init("zhangsan", 18)
	fmt.Printf("s: %v\n", s)
}

输出结果

s: {zhangsan 18}

定义返回方法是否会把初始化的值给返回?

package main

import "fmt"

//定义结构体
type Student struct {
	Name  string
	Age   int
	Score float32
}

//初始化方法
func (s *Student) init(name string, age int, score float32) {
	s.Name = name
	s.Age = age
	s.Score = score
	fmt.Println("初始化完成")
}

//返回结构体
func (s *Student) get() Student {
	return *s
}

func main() {
	var stu Student
	//定义值
	stu.init("zhangsan", 18, 90)
	//返回值
	stu1 := stu.get()
	fmt.Println(stu1)
}

输出结果如下

初始化完成
{zhangsan 18 90}

传统数据类型自定义方法,做数据类型转换

package main

import "fmt"

//别名类型
type integer int

//传统数据类型自定义方法
func (p integer) convert() string {
	return fmt.Sprintf("%d", p)
}

func main() {
	var i integer
	i = 100
	s := i.convert()
	fmt.Printf("类型:%T,值:%s\n", s, s)
}

输出结果如下

类型:string,值:100

指针传入和值传入的区别

值传入不会对数值进行改变,指针传入才可以改变数值

package main

import "fmt"

type integer int

//传统数据类型自定义方法
func (p integer) convert() string {
	return fmt.Sprintf("%d", p)
}

//方法传指针进行数据同步修改
func (p *integer) set(b integer) {
	*p = b
}

func main() {
	var i integer
	i = 100
	s := i.convert()
	fmt.Printf("类型: %T ,值: %s\n", s, s)
	fmt.Printf("类型: %T ,值: %d\n", i, i)
	i.set(200)
	fmt.Printf("i: %v\n", i)
}

输出结果如下

类型: string ,值: 100
类型: main.integer ,值: 100
i: 200

方法继承,组合(匿名字段是组合的特殊形式)

package main

import "fmt"

//父结构体
type Car struct {
	weight int
	name   string
}

//父方法
func (c *Car) Run() {
	fmt.Println("Running")
}

//子结构体Bike
type Bike struct {
	//组合(有名字)
	c     Car
	wheel int
}

//子结构体Train
type Train struct {
	//匿名
	Car
	wheel int
}

func main() {
	var bike Bike
	bike.c.name = "bike"
	bike.c.weight = 500
	bike.wheel = 2

	var train Train
	train.name = "train"
	train.weight = 140000
	train.wheel = 8

	fmt.Println(bike)
	//方法继承,调用父结构体方法
	bike.c.Run()

	fmt.Println(train)
	//方法继承
	train.Run()
}

输出结果如下

{{500 bike} 2}
Running
{{140000 train} 8}
Running

package main

import "fmt"

//父结构体
type Cart struct {
	weight int
	Color  string
}

//父方法
func (c Cart) Run() {
	fmt.Println("Running")
}

//子结构体train
type Train struct {
	Cart
	wheel int
}

//子结构体方法
func (t Train) String() string {
	str := fmt.Sprintf("color:[%s],weight:[%d],wheel:[%d]\n", t.Color, t.weight, t.wheel)
	return str
}

func main() {
	var train Train
	train.Color = "red"
	train.weight = 14000
	train.wheel = 8
	fmt.Println(train)
	train.Run()
	fmt.Printf("%s\n", train)
}

输出结果如下

color:[red],weight:[14000],wheel:[8]
​​​​​​​
Running
color:[red],weight:[14000],wheel:[8]

到此这篇关于Go语言学习之结构体和方法使用详解的文章就介绍到这了,更多相关Go语言结构体 方法内容请搜索易采站长站以前的文章或继续浏览下面的相关文章希望大家以后多多支持易采站长站!

如有侵权,请联系QQ:279390809 电话:15144810328

相关文章

  • golang用什么开发工具?

    golang用什么开发工具?

    golang用的开发工具有:1、Go Revive,是一个Go语言的代码质量检测工具;2、Go Callvis,可以用来可视化Go程序的调用图;3、Gaia,高效,快速,轻量级,并且对开发人员友好。golang用的开发工具有:1、Go Reviverevive 是一个 Go 语言的代码质量检测工具(Linter for Go),具有快速、可配置、可扩展、灵活和美观等特性,可作为 golint 的替
    2020-08-07
  • golang是多线程模式吗?

    golang是多线程模式吗?

    golang是多线程模式的,golang的线程模型是M P G模型,整体上Go程与内核线程是多对多对应的,因此首先来讲就一定是多线程的。golang是多线程模式。 由于gmp中的p与m是将p绑定与m内核线程上,而后p的最大数量有GOPROCESS确定,而M内核线程的数量会由go去限制为10K个,但是由于内核原因做不到这么多,所以这个限制就当做没有吧。拿个图明确一下Golang有些所谓的M比N模型,
    2020-08-07
  • golang用什么开发工具?

    golang用什么开发工具?

    golang用的开发工具有:1、Go Revive,是一个Go语言的代码质量检测工具;2、Go Callvis,可以用来可视化Go程序的调用图;3、Gaia,高效,快速,轻量级,并且对开发人员友好。golang用的开发工具有:1、Go Reviverevive 是一个 Go 语言的代码质量检测工具(Linter for Go),具有快速、可配置、可扩展、灵活和美观等特性,可作为 golint 的替
    2020-08-07
  • golang是单进程的吗?

    golang是单进程的吗?

    golang不是单进程的,而是多线程;golang的线程模型是M P G模型,整体上Go程与内核线程是多对多对应的,因此首先来讲就一定是多线程的。golang不是单进程的,而是多线程。Golang有些所谓的M比N模型,M个线程下可以创建N个go routine,一般而言N远大于M,本质上属于多线程模型,但是协程的调度由Go的runtime决定,强调开发者应该使用channel进行协程之间的同步。至
    2020-08-07
  • 代码详解使用Go基于WebSocket构建视频直播弹幕系统

    代码详解使用Go基于WebSocket构建视频直播弹幕系统

    (1)业务复杂度介绍开门见山,假设一个直播间同时500W人在线,那么1秒钟1000条弹幕,那么弹幕系统的推送频率就是: 500W * 1000条/秒=50亿条/秒 ,想想B站2019跨年晚会那次弹幕系统得是多么的NB,况且一个大型网站不可能只有一个直播间!使用Go做WebSocket开发无非就是三种情况:使用Go原生自带的库,也就是 golang.org/x/net ,但是这个官方库真是出了奇Bu
    2020-08-07
  • PHP语法和Go语法有什么差异?对比介绍

    PHP语法和Go语法有什么差异?对比介绍

    本篇文章给大家对比一下PHP语法和Go语法,带大家了解一下PHP语法和Go语法之间的差异。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。Go 是由 Google 设计的一门静态类型的编译型语言。它有点类似于 C,但是它包含了更多的优点,比如垃圾回收、内存安全、结构类型和并发性。它的并发机制使多核和网络机器能够发挥最大的作用。这是 GoLang 的最佳卖点之一。此外,Go 速度快,
    2020-08-07
  • golang需要什么基础?

    golang需要什么基础?

    golang需要什么基础?golang需要的基础是:Go语言语法特别简单简洁,有C的底子更好,差一些也没关系。前提是你要真心想学,才有足够的动力去学。1、初学Go语言首先弄懂基础语法和概念:基本数据类型、Struct、Array、map、Slice、指针、接口、map、内置函数,常用工具包等,还有接口和Slice的底层数据结构。这些不需要弄特别懂,能自己理解并自己描述我觉得就可以了,关键在实践和应
    2020-08-07
  • golang为什么那么火?

    golang为什么那么火?

    golang为什么那么火?golang那么火的原因:1, Concurrency的原生支持通过语言原生的Goroutine和Channel,很好的支持了Concurrency。你可以把Goroutine理解为非常轻量级的Thread。一个Goroutine只占用2KB的内存,但是一个Thread要占用1MB的内存。Goroutine的创建、销毁和切换的开销,相对于线程来说特别低。你可以随时起上千个
    2020-08-07