简单易懂,解析Go语言中的struct结构体

news2025/2/26 18:59:37

目录

  • 4. struct 结构体
    • 4.1 初始化
    • 4.2 内嵌字段
    • 4.3 可见性
    • 4.4 方法与函数
      • 4.4.1 区别
      • 4.4.2 闭包
    • 4.5 Tag 字段标签
      • 4.5.1定义
      • 4.5.2 Tag规范
      • 4.5.3 Tag意义

4. struct 结构体

go的结构体类似于其他语言中的class,主要区别就是go的结构体没有继承这一概念,但可以使用类型嵌入来实现相似功能。

4.1 初始化

使用type关键字来定义一个新的类型,struct将新类型限定为结构体类型。

结构体中的字段可以为任何类型,但是包含一些特殊类型 如:接口,管道,函数,指针的时候要格外注意

//type定义一个新类型
type newInt int

//type定义一个简单的结构体
type base struct{
   value int   
}

//type定义一个复杂的结构体
type student struct {
   Name string
   age  int
   c    interface{}
   d    func() int
   e    func()
   base     //将base类型嵌入到了student类型中
}

4.2 内嵌字段

内嵌字段大体上有两种方式:显式指定(m1)和隐式指定(m2)

  • 显式指定就相当于把目标结构体当作字段,调用时需要先调用这个字段,在调用目标结构体中的信息
  • 隐式指定相当于把目标结构体中的所有字段都在新结构体中创建了一次,并且指向嵌入结构体内部。同时创建同名嵌入结构体对象[指与base同名]
  • 显式创建同名结构体字段 ≠ 隐式指定
type base struct {
	Value int
}
//显式指定
type m1 struct {
	b base
}

//隐式指定
type m2 struct {
	base
}

//显式指定同名字段
type m3 struct {
   base base
}

对上述结构体进行调用:

  • 只有隐式指定直接操作被嵌入结构体内的数据;
  • 隐式指定后,直接操作嵌入结构体中的数据和通同名结构体操作作用一样
func main() {
	a1 := m1{}
	a2 := m2{}
	a3 := m3{}
	//显式指定只能通过嵌入结构体进行操作
	// a1.Value = 1 //a1.Value undefined (type m2 has no field or method Value)
	a1.b.Value = 2
	//隐式指定两种操作数据方法操作的是同一个变量
	a2.Value = 2
	a2.base.Value = 3
	fmt.Println(a2.Value) //3
	//显式指定同名变量 ≠ 隐式指定 
	// a3.Value = 3 //a3.Value undefined (type m3 has no field or method Value)
	a3.base.Value = 4
}

当内嵌字段中的字段与结构体中得字段同名时:

  • 直接调用时是指定当前结构体中显式定义的字段,但嵌入结构体中的字段仍可通过嵌入类型进行调用
  • 方法同理
//数据
func main() {
	a1 := m1{}
	a1.Value = "hello world"
	a1.base.Value = 1
	fmt.Println(a1)
	//获取a1中的所有字段类型
	t := reflect.TypeOf(a1)
	for i := 0; i < t.NumField(); i++ {
		field := t.Field(i)
		fieldType := field.Type
		fmt.Printf("Field: %s, Type: %s\n", field.Name, fieldType.Name())
	}
}

type base struct {
	Value int
}

type m1 struct {
	Value string
	base
}
/*
{hello world {1}}
Field: Value, Type: string
Field: base, Type: base
 */
//方法
func main() {
   a1 := m1{}
   a1.test()
   a1.base.test()
   fmt.Println(a1)
   //获取a1中的所有字段类型
   t := reflect.TypeOf(a1)
   for i := 0; i < t.NumField(); i++ {
      field := t.Field(i)
      fieldType := field.Type
      fmt.Printf("Field: %s, Type: %s\n", field.Name, fieldType.Name())
   }
}

type base struct {
   Value int
}

func (b *base) test() {
   b.Value = 1
}
func (m *m1) test() {
   m.Value = "hello world"
}

type m1 struct {
   Value string
   base
}
/*
{hello world {1}}
Field: Value, Type: string
Field: base, Type: base
 */

4.3 可见性

  • 首字母大写表示该字段/方法/结构体为可导出的,反之为不可导出的
  • 在同一个包内,不区分是否可导出,都可访问;包外只能访问可导出的字段/方法/结构体
  • 不可导出的字段不可以进行序列化(转化为json)
  • 可通过可导出的方法去操作不可导出的字段
// test/test1.go
package test

type User struct {
   Name string
   age  int
}

func (u *User) Test() {
   u.Name = "hello"
   u.age = 18
}

func (u *User) test() {
   u.Name = "world"
   u.age = 81
}

type student struct {
   Name string
   age  int
}
//main.go
package main

import (
   "fmt"
   "test/test"
)

func main() {
   a := test.User{}
   //b := test.student{} // 不能在包外访问未导出结构体
   a.Name = "123"
   //a.age = 123 // 不能在包外访问未导出字段
   a.Test()
   //a.test() // 不能在包外访问未导出方法
   fmt.Println(a)
   /*
      {hello 18}
    */
}

4.4 方法与函数

4.4.1 区别

  • 方法定义是必须有一个接收器(receiver);函数不需要
  • 大部分情况下,方法的调用需要有一个对象;函数不需要
  • 由于go中的所有传递都是值传递,也就是将数据复制一份再调用,所以如果想要修改原本对象的值,就要传递指针,进行引用传递
func 函数名 (参数) 返回值类型 {函数体} //函数定义
func (接收器) 方法名  (参数) 返回值类型 {函数体} // 方法定义
func main() {
	a := User{}
	a.setName() // 方法调用
	fmt.Println(a)
	setName(&a) // 函数调用
	fmt.Println(a)
}
/*
   {world 0}
   {hello 0}
 */

type User struct {
	Name string
	age  int
}
//函数
func setName(u *User) {
	u.Name = "hello"
}
//方法
func (u *User) setName() {
	u.Name = "world"
}

值传递时可以通过返回值的方式修改目标对象

func main() {
	a := User{}
	a = a.setName() //需要显式给原变量赋值
	fmt.Println(a)
	a = setName(a)  //需要显式给原变量赋值
	fmt.Println(a)
}

type User struct {
	Name string
	age  int
}

func setName(u User) User {
	u.Name = "hello"
	return u
}
func (u User) setName() User {
	u.Name = "world"
	return u
}

4.4.2 闭包

说到函数和方法,就必须说一下闭包

什么是闭包?

简单来说,就是函数内部引用函数外部变量,导致变量生命周期发生变化。这样的函数就叫做闭包

常见于函数返回值为另一个函数时

package main

import "fmt"

func main() {
   b := test()
   fmt.Println(b())
   fmt.Println(b())
}

func test() func() int {
   a := 1
   return func() int {
      a++
      return a
   }
}

上面的函数导致变量a无法正常释放,导致变量逃逸

go build -gcflags="-m" main.go
# command-line-arguments
./main.go:11:6: can inline test
./main.go:13:9: can inline test.func1
./main.go:6:11: inlining call to test
./main.go:13:9: can inline main.test.func1
./main.go:7:15: inlining call to main.test.func1
./main.go:7:13: inlining call to fmt.Println
./main.go:8:15: inlining call to main.test.func1
./main.go:8:13: inlining call to fmt.Println
./main.go:6:11: func literal does not escape
./main.go:7:13: ... argument does not escape
./main.go:7:15: ~R0 escapes to heap
./main.go:8:13: ... argument does not escape
./main.go:8:15: ~R0 escapes to heap
./main.go:12:2: moved to heap: a
./main.go:13:9: func literal escapes to heap

4.5 Tag 字段标签

4.5.1定义

在reflect包中提供了获取字段名称、类型、Tag的方法(上文展示过获取名称和类型)

结构体StructField表示结构体的一个字段(reflect/type.go)

// A StructField describes a single field in a struct.
type StructField struct {
	// Name is the field name.
	Name string

	// PkgPath is the package path that qualifies a lower case (unexported)
	// field name. It is empty for upper case (exported) field names.
	// See https://golang.org/ref/spec#Uniqueness_of_identifiers
	PkgPath string

	Type      Type      // field type
	Tag       StructTag // field tag string
	Offset    uintptr   // offset within struct, in bytes
	Index     []int     // index sequence for Type.FieldByIndex
	Anonymous bool      // is an embedded field
}

type StructTag string 

4.5.2 Tag规范

StructTag本质上就是字符串,理论上任何形式都符合规范。但通常情况下约定,Tag的格式应该是key:“value”

  • key:非空字符串,不能包含控制字符,空格,引号,冒号
  • value:双引号包围的字符串
  • 冒号前后不能有空格,多个value用逗号隔开,key之间用空格隔开
  • key一般表示用途,value表示控制指令;

4.5.3 Tag意义

  • Go语言反射机制可以给结构体成员赋值,用Tag可以决定赋值的动作
  • 可以使用定义好的Tag规则,参考规则就可以继续不同的操作
//仅对Tag值为true的字段赋值(Tag决定赋值动作)
type Person struct {
	Name string `assign:"true"`
	Age  int    `assign:"false"`
}

func assignValues(v interface{}) {
	val := reflect.ValueOf(v).Elem() // 获取指针指向的值
	typ := val.Type()

	for i := 0; i < val.NumField(); i++ {
		field := val.Field(i)
		tag := typ.Field(i).Tag.Get("assign") // 获取字段的tag

		if tag == "true" {
			// 根据字段类型赋值
			switch field.Kind() {
			case reflect.String:
				field.SetString("Default Name")
			case reflect.Int:
				field.SetInt(25)
			}
		}
	}
}

func main() {
	p := &Person{}
	assignValues(p)
	fmt.Printf("Person: %+v\n", p)
}
/*
Person: &{Name:Default Name Age:0}
 */

下方例子使用了json:“kind,omitempty”,这个tag规定了字段为空不进行序列化;

import (
	"encoding/json"
	"fmt"
)

type Person struct {
	Name  string `json:"kind,omitempty"` //为空时不进行序列化
	Value int
	age   int
}

func main() {
	// 创建一个 Person 实例
	p := Person{Name: "", Value: 100, age: 100}

	// 序列化为 JSON
	jsonData, _ := json.Marshal(p)
	fmt.Println(string(jsonData)) 
	/*
	{"Value":100}
	 */
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2306520.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

国产编辑器EverEdit - 洞察秋毫!文件比较功能!

1 文件比较 1.1 应用场景 项目开发过程中&#xff0c;可能不同的部分会由不同的人在负责&#xff0c;存在一个文件多人编辑的情况&#xff0c;用户需要寻找差异&#xff0c;并将文档进行合并&#xff0c;比较专业的文本比较工具为BeyondCompare&#xff0c;WinMerge等。   如…

QARepVGG--含demo实现

文章目录 前言引入Demo实现总结 前言 在上一篇博文RepVGG中&#xff0c;介绍了RepVGG网络。RepVGG 作为一种高效的重参数化网络&#xff0c;通过训练时的多分支结构&#xff08;3x3卷积、1x1卷积、恒等映射&#xff09;和推理时的单分支合并&#xff0c;在精度与速度间取得了优…

kotlin 知识点 七 泛型的高级特性

对泛型进行实化 泛型实化这个功能对于绝大多数Java 程序员来讲是非常陌生的&#xff0c;因为Java 中完全没有这个概 念。而如果我们想要深刻地理解泛型实化&#xff0c;就要先解释一下Java 的泛型擦除机制才行。 在JDK 1.5之前&#xff0c;Java 是没有泛型功能的&#xff0c;…

Transformer LLaMA

一、Transformer Transformer&#xff1a;一种基于自注意力机制的神经网络结构&#xff0c;通过并行计算和多层特征抽取&#xff0c;有效解决了长序列依赖问题&#xff0c;实现了在自然语言处理等领域的突破。 Transformer 架构摆脱了RNNs&#xff0c;完全依靠 Attention的优…

Qt学习 网络编程 TPC通信

一 基本网络端口 1 网络编程基本概念 通讯方式&#xff1a;信息的通讯时通过网络来进行&#xff0c;通讯方式有两种&#xff0c;TCP和UDP通信&#xff0c;TCP通讯是专用通道&#xff0c;指定某个信息只能走某个通道&#xff0c;UDP则是非专用通道&#xff0c;比如一个车队&am…

ESP32-S3 实战指南:BOOT-KEY 按键驱动开发全解析

一、基础知识 本篇我们使用 BOOT 按键来学习一下 GPIO 功能&#xff0c;首先补充一下相关术语介绍。 1、GPIO&#xff08;General Purpose Input/Output&#xff09; GPIO 是微控制器上的通用引脚&#xff0c;既可以作为输入&#xff08;读取外部信号&#xff09;&#xff0…

ssh配置 远程控制 远程协作 github本地配置

0.设备版本 windows11 ubuntu24.0.4 1.1 在 Linux 上启用 SSH 服务 首先&#xff0c;确保 Linux 计算机上安装并启用了 SSH 服务。 安装和启动 OpenSSH 服务&#xff08;如果未安装&#xff09; # 在终端安装 OpenSSH 服务&#xff08;如果尚未安装&#xff09; sudo apt …

C++知识整理day9——继承(基类与派生类之间的转换、派生类的默认成员函数、多继承问题)

文章目录 1.继承的概念和定义2.基类与派生类之间的转换3.继承中的作用域4.派生类的默认成员函数5.实现一个不能被继承的类6.继承与友元7.继承与静态成员8.多继承和菱形继承问题8.1 继承分类及菱形继承8.2 虚继承 1.继承的概念和定义 概念&#xff1a; 继承(inheritance)机制是⾯…

2024年国赛高教杯数学建模A题板凳龙闹元宵解题全过程文档及程序

2024年国赛高教杯数学建模 A题 板凳龙闹元宵 原题再现 “板凳龙”&#xff0c;又称“盘龙”&#xff0c;是浙闽地区的传统地方民俗文化活动。人们将少则几十条&#xff0c;多则上百条的板凳首尾相连&#xff0c;形成蜿蜒曲折的板凳龙。盘龙时&#xff0c;龙头在前领头&#x…

华为认证考试证书下载步骤(纸质+电子版)

华为考试证书可以通过官方渠道下载相应的电子证书&#xff0c;部分高级认证如HCIE还支持申请纸质证书。 一、华为电子版证书申请步骤如下&#xff1a; ①访问华为培训与认证网站 打开浏览器&#xff0c;登录华为培训与认证官方网站 ②登录个人账号 在网站首页&#xff0c;点…

【Android】用 chrome://inspect/#devices 调试H5页面

通常做Android开发的过程中&#xff0c;不可避免的需要遇到去与H5交互&#xff0c;甚至有时候需要去调试H5的信息。 这里分享一下Android工程里如何调试H5页面信息&#xff1a; 直接在浏览器地址栏输入 &#xff1a; chrome://inspect/#devices 直接连接手机usb,打开开发者模式…

贪心算法精品题

1.找钱问题 本题的贪心策略在于我们希望就可能的保留作用大的5元 class Solution { public:bool lemonadeChange(vector<int>& bills) {std::map<int ,int> _map;for(auto ch:bills){if(ch 5) _map[ch];else if(ch 10){if(_map[5] 0) return false;else{_m…

WEB1~6通杀

##解题思路 这六道题&#xff0c;通杀了&#xff0c;只因为是PHP的特性 来&#xff0c;看web6&#xff0c;过滤最复杂的正则&#xff0c;而且不能解析成大于999的值&#xff0c;但是&#xff0c;php是弱类型的语言&#xff0c;我只要输入任意字符数字&#xff0c;最终值就为0&…

孜然单授权系统V2.0PHP授权系统

孜然单授权V1.0系统&#xff0c;延续了2022年开发的孜然多应用授权系统V2.0 变更&#xff1a;多应用变单系统&#xff0c;去除没用的垃圾代码&#xff0c;从0开发&#xff0c;去除了一些没用的功能 完善了开发文档&#xff0c;之前那套是我写着玩的屎山代码&#xff0c;V1.0将展…

Apache SeaTunnel 构建实时数据同步管道(最新版)

文章作者 王海林 白鲸开源 数据集成引擎研发 Apache SeaTunnel Committer & PMC Member&#xff0c;Apache SkyWalking Committer&#xff0c;多年平台研发经验&#xff0c;目前专注于数据集成领域。 导读 在当今数字化快速发展的时代&#xff0c;数据已然成为企业决策…

服务器离线部署DeepSeek

目标 本次部署的目标是在本地服务器上部署DeepSeek。但是该服务不能连接外网&#xff0c;因此只能使用离线部署的方式。为了一次完成部署。现在云服务器上进行尝试。 云服务器部署尝试 云服务器配置 CentOS72080Ti 11GB 安装准备 1、上传iso并配置为本地yum源 安装前先将…

ComfyUI:Stable Diffusion 及 LoRA、VAE 、ControlNet模型解析

目录 Stable Diffusion流程 扩散过程 去噪过程 checkpoints LoRA LoRA 位置与结构 LoRA 层与原层的关系 LoRA 层的参数拆解 VAE 训练特定 VAE 时更新的参数部分 ControlNet ControlNet 位置与结构 ControlNet 的训练过程 ControlNet 的参数处理与信息融合 Contr…

微信小程序:多菜单栏设计效果

一、实现效果 二、代码 wxml 编辑前端界面,步骤 菜单逻辑: 逐步取出数组中的项,首先取出顶部菜单项,然后选中后取出选中的底部数据(左侧菜单+右侧内容),然后点击左侧菜单取出选中的左侧菜单对应的右侧内容 ①这里我的数据是全部封装到一个数组对象的,首先我的循环…

【Linux Oracle】time命令+oracle exp压缩

Linux && Oracle相关文档&#xff0c;希望互相学习&#xff0c;共同进步 风123456789&#xff5e;-CSDN博客 1.说明 Linux中的time命令&#xff1a;主要用于测量命令的执行时间&#xff0c;并显示该命令在执行过程中所使用的系统资源情况&#xff0c;如CPU时间、内存和…

20分钟 Bash 上手指南

文章目录 bash 概念与学习目的第一个 bash 脚本bash 语法变量的使用位置参数管道符号&#xff08;过滤条件&#xff09;重定向符号条件测试命令条件语句case 条件分支Arrayfor 循环函数exit 关键字 bash 脚本记录历史命令查询文件分发内容 bash 概念与学习目的 bash&#xff0…