Go语言结构体struct详解,Go空结构体的这些妙用你知道吗?
本文详解了Go语言结构体的各个知识点,最后介绍了空结构体的3种妙用。希望对你有帮助。
定义
结构体,是一种自定义的数据类型,由多个数据类型组合而成。用于描述一类事物相关属性。
定义方式:
type 类型名 struct {字段名 字段类型…
}//示例:
type Animal struct {Name stringAge int
}
实例化
结构体和结构体指针,两者的实例化有所区别
提供多种写法,灵活使用:
//结构体实例化
//写法1
//var a Animal
//a.Name = "aaa"
//a.Age = 18
//写法2
a := Animal{ Name: "dog",Age: 18,
}
fmt.Println(fmt.Sprintf("%T - %v", a, a)) //main.Animal - {dog 18}//结构体指针实例化
//写法1
var b *Animal
b = new(Animal)
//写法2
//b := new(Animal)
//写法3
//b := &Animal{}
b.Name = "cat" //在底层是(*b).Name = "cat",这是Go语言帮我们实现的语法糖
fmt.Println(fmt.Sprintf("%T - %v", b, b)) //*main.Animal - &{cat 0}
注意:结构体指针必须手动初始化,分配内存地址
匿名结构体
适用于临时数据存储的场景
var v struct {Name stringAge int
}
fmt.Println(v)
空结构体
不占用内存空间
var v struct{}
fmt.Println(unsafe.Sizeof(v)) //0v1 := struct{}{}
fmt.Println(unsafe.Sizeof(v1)) //0
构造函数
Go没有自带的构造函数,采用自实现
方式1
结构体不复杂,可以返回结构体类型,值拷贝性能开销小
func NewPerson(name string, age int8) Person {return Person{name: name,age: age,}
}
定义方式2
结构体复杂,得返回结构体指针类型,避免值拷贝产生的性能开销
func NewPerson(name string, age int8) *Person {return &Person{name: name,age: age,sex: sex,country:country,province:province,city:city,town:town,address:address,}
}
方法与接收者
方法(Method)是一种作用于特定类型变量的函数。这种特定类型变量叫做接收者(Receiver)。接收者的概念就类似于PHP中的this或者 self。
方法与函数区别:函数不属于任何类型,方法属于特定类型。函数没有接收者,方法有接收者。
标准格式:
func (接收者变量 接收者类型) 方法名(参数列表) (返回参数) {函数体
}
接收者类型(两种):
- 非指针类型:发生值拷贝产生副本,方法内修改字段,只在方法内生效;
- 指针类型:不产生副本,方法内修改字段,同步生效;
func NewPerson(name string, age int8) *Person {return &Person{name: name,age: age,}
}func (p *Person) Dream() {p.name = "aaa"fmt.Printf("%s的梦想是学好Go语言\n", p.name) //aaa的梦想是学好Go语言
}func main() {p1 := NewPerson("小王子", 25)p1.Dream()fmt.Println(p1) //&{aaa 25}
}
什么时候使用指针类型的接收者:
- 需要修改接收者中的值
- 接收者是拷贝代价比较大的大对象
- 保证一致性,在同一个文件中,如果有某个方法使用了指针接收者,那么其他的方法也建议使用指针接收者
注意点:
1.接收者类型,可以是任何类型,不仅仅只针对结构体类型。但要注意下,类型和方法定义需要在同一个包下面
type MyInt intfunc (i MyInt) SayInt() {fmt.Println("my type is MyInt")
}func main() {var i1 MyInti2 := MyInt(10)i1.SayInt()i2.SayInt()
}输出结果:
my type is MyInt
my type is MyInt
匿名字段
结构体允许其成员字段在声明时没有字段名而只有类型,这种没有名字的字段就称为匿名字段
type User struct {Name stringGender stringAddress //匿名字段
}type Address struct {Province stringCity stringCreateTime string
}func main() {var u1 Useru1.Name = "张三"u1.Gender = "男"u1.Address.City = "北京" //匿名字段默认使用类型名作为字段名u1.CreateTime = "2019" //匿名字段可以省略,但注意多个匿名字段下有相同字段名,会编译失败,所以建议不采用省略写法fmt.Println(u1)
}
但需要注意字段名冲突问题,所以不建议使用省略写法操作匿名字段
实现面向对象的“继承”特性
Go不是面向对象编程的语言,但可以通过嵌套结构体的方式,来实现面向对象的“继承”特性
type Animal struct {Name stringAge int
}func (a Animal) Say() {fmt.Println(fmt.Sprintf("1-my name is %s and age is %d", a.Name, a.Age))
}type Cat struct {Animal //嵌套结构体实现继承
}func main() {c1 := Cat{}c1.Name = "加菲猫"c1.Age = 5c1.Say()//输出结果://1-my name is 加菲猫 and age is 5
}
子类还可以重写父类的Say方法,并且还能拥有自己的Run方法:
func (c Cat) Say() {fmt.Println(fmt.Sprintf("2-my name is %s and age is %d", c.Name, c.Age))
}func (c Cat) Run() {fmt.Println(fmt.Sprintf("my name is %s,还是跑步高手", c.Name))
}func main() {c1 := Cat{}c1.Name = "加菲猫"c1.Age = 5c1.Say()c1.Run()//输出结果://2-my name is 加菲猫 and age is 5//my name is 加菲猫,还是跑步高手
}
标签tag
通过反射机制,识别结构体的标签,容错能力较差,需要注意使用
标准格式:
`key1:"value1" key2:"value2"`
使用注意事项:
- 外层使用
反引号包起来,里边value需要使用双引号包起来; - KV之间使用
冒号,多个KV之间使用空格; (注意:冒号前后不要加其他符号)
使用示例:
goframe v2的标准路由注册就是使用标签tag的方式定义的,大家感兴趣可以看下我们开源项目的代码:
https://github.com/wangzhongyang007/goframe-shop-v2
结构体与JSON系列化
给结构体添加json标签,然后做json序列化操作:
- 首字母大写字段(公开) :会转换成json标签指定的字段名,若未指定,则使用自身字段名;
- 首字小写字段(私有) :不会输出,因为这类字段仅在定义当前结构体的包中可访问;
简单示例:
type CardInfo struct {Title string `json:"title"`Desc stringheight int `json:"height"`
}func main() {c1 := CardInfo{Title: "成长之星",Desc: "balabala",height: 100,}data, _ := json.Marshal(c1)fmt.Println(string(data)) //{"title":"成长之星","Desc":"balabala"}str := "{"title":"title111", "desc":"desc222", "height":20}"c2 := CardInfo{}_ = json.Unmarshal([]byte(str), &c2)fmt.Println(c2) //{title111 desc222 0}
}
空结构体
上文为大家简单介绍了空结构体,使用unsafe.SizeOf()方法,明确知道了空结构体,它不占用存储空间。
(即“宽度”为0,宽度描述了一个类型的实例所占用的存储空间的字节数)
s := struct{}{}
fmt.Println(unsafe.Sizeof(s)) //0
在项目代码中,我们经常都会看到空结构体struct{}{}的使用,它有什么作用,适合什么场景使用呢?
空结构体作用
请大家注意:结构体包含一个指针和指针指向的数据,下文所说的不占用内存其实指的是指针指向的数据为null,但是空结构体最为一个变量它的指针肯定是占用内存空间的,只是单用很小。
因为空结构体的值不占据内存空间的特性,因此被广泛作为各种场景下的占位符使用。
- 一是节省资源
- 二是空结构体本身就具备很强的语义:即这里不需要任何值,仅作为占位符。
空结构体使用场景
主要使用场景有3个:
- 实现集合类型
- 实现空通道
- 实现方法接收者
下面逐个为大家详解
1.实现集合类型
Go语言本身是没有集合类型(Set),通常是使用map来替代。
但有个问题:就是集合类型,只需要用到key(键),不需要用到value(值)
如果value使用bool来表示,实际会占用1个字节的空间,为了节省空间,这时空结构体就可以大显身手了
type Set map[int]struct{}func main() {s := make(Set)s.add(1)s.add(2)s.add(3)s.remove(2)fmt.Println(s.exist(1))fmt.Println(s)//输出://true//map[1:{} 3:{}]
}
func (s Set) add(num int) {s[num] = struct{}{}
}
func (s Set) remove(num int) {delete(s, num)
}
func (s Set) exist(num int) bool {_, ok := s[num]return ok
}
空结构体作为占位符,不会额外增加不必要的内存开销,很方便的就把问题给解决了
2.实现空通道
在Go语言 channel的使用场景中,常常会遇到通知型 channel,其不需要发送任何数据,只是用于协调 Goroutine 的运行,用于流转各类状态或是控制并发情况。
这类情况就特别适合使用空结构体,只做个占位,不浪费内存空间
func main() {ch := make(chan struct{})go worker(ch)// Send a message to a worker.ch <- struct{}{}// Receive a message from the worker.<-chprintln("AAA")//输出://BBB//AAA
}func worker(ch chan struct{}) {// Receive a message from the main program.<-chprintln("BBB")// Send a message to the main program.close(ch)
}
由于该 channel 使用的是空结构体,因此也不会带来额外的内存开销
3.实现方法接收者
使用结构体类型的变量作为方法接收者,有时结构体可以不包含任何字段属性。这种情况,可以用int或者string来替代,但它们都会占用内存空间,所以使用空结构体是比较合适的。
并且也有利于未来针对该类型进行公共字段等的增加,容易扩展和维护
type T struct{}func methodUse() {t := T{}t.Print()t.Print2()//输出://哈哈哈Print//哈哈哈Print2
}func (t T) Print() {fmt.Println("哈哈哈Print")
}
func (t T) Print2() {fmt.Println("哈哈哈Print2")
}
总结
本文详解了Go语言结构体的各种知识点,最后针对空结构体的作用和使用场景,进行了详细的讲解。在之后的实际项目开发过程中,只用占位不用实际含义,那么我们就都可以使用空结构体,可以极大的节省不必要的内存开销。
希望对大家有帮助,兄弟们觉好留言哦。
坚持写作
这篇文章来自知识星球中劲仔的投稿,欢迎加入我们,坚持写作输出,一起成长进步。
相关文章:
Go语言结构体struct详解,Go空结构体的这些妙用你知道吗?
本文详解了Go语言结构体的各个知识点,最后介绍了空结构体的3种妙用。希望对你有帮助。 定义 结构体,是一种自定义的数据类型,由多个数据类型组合而成。用于描述一类事物相关属性。 定义方式: type 类型名 struct {字段名 字段类…...
华为OD机试 - 航天器(Python) | 机试题+算法思路+考点+代码解析 【2023】
航天器 题目 给航天器一侧加装长方形和正方形的太阳能板(图中的斜线区域); 需要先安装两个支柱(图中的黑色竖条); 再在支柱的中间部分固定太阳能板; 但航天器不同位置的支柱长度不同; 太阳能板的安装面积受限于最短一侧的那支支柱的长度; 现提供一组整型数组的支柱高度数据;…...
【Optional】告别丑陋判空,使用Optional类
一、概述 当项目中充斥着大量的、丑陋的判空语句,如下: if (user ! null) {Address address user.getAddress();if (address ! null) {Country country address.getCountry();if (country ! null) {String isocode country.getIsocode();if (isocod…...
魔兽世界服务端端新手搭建教程
明杰也是很久以前开始研究魔兽服务器架设,主要原因是亚服经常要排队6-7个小时,去不排除的服和单机没啥区别,以怀旧服玩到10级以后就开始玩335端,一开始也和新入手的人一样云里雾里的,经过长时间的学习总算有点成就,向新…...
宝塔搭建实战人才求职管理系统mobile手机端vue源码(五)
大家好啊,我是测评君,欢迎来到web测评。 上一期给大家分享骑士cms会员管理member前端vue在本地运行打包、宝塔发布部署的方式,本期给大家分享,mobile移动端vue怎么在本地运行,打包,实现线上功能更新替换的方…...
生态应用:探讨 NGINX 与上下游系统集成时的开发经验
NGINX 作为一款高性能的 Web 服务器和反向代理服务器,在各种应用场景中广泛应用。随着业务的发展,我们在使用 NGINX 时,可能需要将其与其他系统进行集成,以实现更加复杂的业务需求。 本文将结合实际应用场景,探讨 NGI…...
ArcGIS批量拼接大量栅格遥感影像:Mosaic工具
本文介绍在ArcGIS下属的ArcMap软件中,基于Mosaic工具,批量对大量栅格遥感影像文件加以拼接、镶嵌的方法。 在GIS应用中,我们时常需要对大量栅格遥感影像数据加以批量拼接的工作。这一步骤可以基于Python语言实现,具体可以参考文章…...
Flink UI部署jar包报错
错误描述: 通过Flink的UI中的Submit New Job菜单添加jar包的时候提示报错。报错信息的关键字是“The LocalStreamEnvironment cannot be used when submitting a program through a client, or running in a TestEnvironment context”,最关键的是“Loc…...
Linux就该这么学:RAID与LVM磁盘阵列技术
这里写目录标题 7.1.1 部署磁盘阵列7.1.2 损坏磁盘阵列及修复7.1.3 磁盘阵列+备份盘7.1.4 删除磁盘阵列1. 需要将所有的磁盘都设置成停用状态:2. 逐一移除出去3. 续停用整个RAID磁盘阵列7.2 LVM逻辑卷管理器7.2.1 部署逻辑卷7.2.2 扩容逻辑卷7.2.3 缩小逻辑卷7.2.4 逻辑卷快照…...
Prometheus+Grafana监控
1、简介1.1 Prometheus官网地址:https://prometheus.io/Prometheus是一个开源的监控系统,起源于SoundCloud。它由以下几个核心组件构成:数据爬虫: 根据配置的时间定期的通过HTTP抓去metrics数据。time-series 数据库: …...
【JUC2022】第三章 线程中断与 LockSupport
【JUC2022】第三章 线程中断与 LockSupport 文章目录【JUC2022】第三章 线程中断与 LockSupport一、线程中断1.什么是中断机制2.中断 API3.代码实现4.Thread.sleep()二、LockSupport1.什么是 LockSupport2.代码实现3.总结一、线程中断 1.什么是中断机制 首先,一个…...
数据结构刷题(七):202快乐数、1两数之和、454四数相加II、15三数之和、18四树之和
1.快乐数题目链接思路:使用set,当set中出现相同元素时,返回false注意:while循环中语句顺序; 除数取余。解法:public boolean isHappy(int n){Set<Integer> res new HashSet<>();int[] arr ne…...
华为机试题:HJ80 整型数组合并(python)
文章目录知识点详解1、input():获取控制台(任意形式)的输入。输出均为字符串类型。 1.1、int(input()) 与 map(int, input().spilt()) 的区别 1.2、input() 与 list(input()) 的区别、及其相互转换方法2、print() :打印输出…...
spring boot——自定义依赖实现自动配置
需求 要实现的功能是:实现一个可以支持miniooss两种方式,上传下载文件的自定义依赖。其中还包括一些创建桶、删除桶、删除文件等功能,但是最主要的是实现自动配置。 如果对spring理解很深的话,自动配置这些东西很容易理解&#…...
QMap 判断是否value是否已经存在,结合Sleep函数测试
网上查了资料,基本说的都是通过.value判断是否已经之前的key值,但是尝试.了一下发现有.key的函数,对比着来就感觉这个函数是用来判断是否已经存在value值,于是开始百度也几乎没有找到相关资料,只好自己看官方文档&…...
vue后台管理系统项目-table选择多行数据分页列表、一键全选重置功能
table选择多行数据 功能介绍: 1.列表分页功能; 2.一键全选,选中列表所有数据; 3.全选,选中当前页数据; 4.重置,清除选中状态; 5.列表搜索查询; 效果: 1.列表分…...
论文解读 | [CVPR2019] 基于自适应文本区域表示的任意形状场景文本检测
目录 1 研究背景及意义 2 总体设计 3 方法论 3.1 自适应文本区域表示 3.2 文本建议 3.3 建议改进 4 损失函数 5 实验及结果 1 研究背景及意义 现有的场景文本检测方法使用固定点数的多边形来 表示文本区域。例如,水平文本使用2个点(左上/右下)表示文本区域&…...
2月编程语言排行榜谁还没有看?
近日,TIOBE公布了2023年2月编程语言排行榜,本月各个语言表现如何?谁又摘得桂冠?一起来看看吧! TIOBE 2月Top15编程语言: 详细榜单查看TIOBE官网 https://www.tiobe.com/tiobe-index/ 关注IT行业的小伙伴们…...
nginx.conf配置方法详细介绍
从前面的内容学习中,我们知道Nginx的核心配置文件默认是放在/usr/local/nginx/conf/nginx.conf,这一节,我们就来学习下nginx.conf的内容和基本配置方法。读取Nginx自带的Nginx配置文件,我们将其中的注释部分【学习一个技术点就是在…...
【微信小程序】一文带你吃透开发中的常用组件
写在前面 小程序中的组件也是由宿主环境提供的,开发者可以基于组件快速搭建出漂亮的页面结构。 官方把小程序的组件分为了9大类,分别是: 1.视图容器 2.基础内容 3.表单组件 4.导航组件 5.媒体组件 6.地图组件 7.画布组件 …...
Unity Il2CppDumper原理与实战:解析元数据与二进制对齐
1. 这不是“破解工具”,而是Unity开发者该懂的二进制真相课 你刚在Unity Asset Store下载了一个功能惊艳的插件,却在打包iOS后发现部分逻辑失效;或者接手一个没有源码的旧项目,只有一堆 .dll 和 .so 文件,连主入口…...
DeepSeek代码质量评估实战手册:7步完成从混沌到可度量的质变跃迁
更多请点击: https://kaifayun.com 第一章:DeepSeek代码质量评估的底层逻辑与核心价值 DeepSeek代码质量评估并非简单地统计行数或检测语法错误,而是基于多维语义理解构建的推理系统。其底层逻辑融合了静态分析、符号执行与大语言模型生成式…...
Hitboxer:开源SOCD清理工具,3分钟提升游戏操作精准度
Hitboxer:开源SOCD清理工具,3分钟提升游戏操作精准度 【免费下载链接】socd Key remapper for epic gamers 项目地址: https://gitcode.com/gh_mirrors/so/socd 你是否在激烈的游戏对抗中经历过这样的挫败:同时按下左右方向键时角色卡…...
ARM PMU性能监控单元原理与实践指南
1. ARM PMU性能监控单元概述性能监控单元(PMU)是现代ARM处理器中用于硬件级性能分析的核心组件。它通过一组可编程的硬件计数器,实现对处理器内部各种关键事件的精确测量。这些事件涵盖了从指令执行、缓存访问到内存子系统行为等处理器活动的…...
基于声卡与电流互感器的安全交流功率测量系统设计与实践
1. 项目概述:用声卡安全测量交流功率我一直对各种测量技术抱有浓厚的兴趣,毕竟“测量即认知”这句老话在今天依然适用。对于电力消耗和产出,没有什么比直接测量更能说明问题了。交流功率的测量,核心在于同时获取电压和电流的瞬时值…...
Windows10下V-REP教育版安装保姆级教程(附百度网盘资源与避坑点)
Windows10系统V-REP教育版完整安装指南:从下载到实战避坑在机器人仿真和自动化控制领域,V-REP(现更名为CoppeliaSim)作为一款功能强大的跨平台机器人仿真软件,已经成为众多工科学生和研究人员的首选工具。特别是其教育…...
危急时刻的六条基本安全提示
人机协作,AI模型:Deepseek 仅供参考 危急时刻的六条基本安全提示 以下内容仅为通用性安全建议,供在紧急情况下保持冷静、保护自身安全时参考。所有建议均基于常理和公共安全常识,不包含任何具体操作细节或可能被不当使用的信息…...
DeepSeek模型微调全链路解析:从数据准备、LoRA配置到推理部署的7大关键步骤
更多请点击: https://intelliparadigm.com 第一章:DeepSeek模型微调全链路概览 DeepSeek系列大语言模型(如DeepSeek-V2、DeepSeek-Coder)凭借其开源特性、高性能推理能力与丰富的领域适配性,已成为工业界与学术界微调…...
国产麒麟系统上编译GDAL 3.2.1踩坑记:从PROJ6依赖缺失到Qt环境集成
麒麟系统GDAL 3.2.1编译实战:PROJ6依赖修复与Qt工程深度集成在国产操作系统生态中部署地理数据处理工具链,往往会遇到比常规Linux发行版更复杂的依赖问题。最近在麒麟系统上为北斗定位项目编译GDAL 3.2.1时,遭遇了经典的"PROJ 6 symbols…...
实战解锁:在Blender中掌握专业级MMD动画制作全流程
实战解锁:在Blender中掌握专业级MMD动画制作全流程 【免费下载链接】blender_mmd_tools MMD Tools is a blender addon for importing/exporting Models and Motions of MikuMikuDance. 项目地址: https://gitcode.com/gh_mirrors/bl/blender_mmd_tools MMD …...
