十一、快速入门go语言之接口和反射

文章目录

    • 接口
      • :one: 接口基础
      • :two: 接口类型断言和空接口
        • :star2: 空接口实现存储不同数据类型的切片/数组
        • :star2: 复制切片到空接口切片
        • :star2: 类型断言
    • 反射

📅 2024年5月9日

📦 使用版本为1.21.5

接口

十、Java类的封装和继承、多态 - 七点半的菜市场 (tanc.fun) Java的接口

ps: 我感觉Go的接口和类方法两个就很模糊

1️⃣ 接口基础

⭐️ 接口就是一些未实现功能的集合(我是这样理解的),为了实现多态(就是多状态)

type PersonIF interface {Status() //人类的状态Skills() string //会的技能
}

⭐️ 在Go语言中接口可以有值,它的值默认是nil,接口本质上是一个指针,一个类型如果实现了接口中的所有方法,那么这个类实现的方法就会使用指针自动指向接口的方法

⚠️ 注意是所有抽象方法,如果有一个没实现都会报错

⭐️ 类型不需要使用什么关键字来表面实现了这个接口,多个类型可以同时实现一个接口,一个类型也可以实现多个接口

type PersonIF interface {Status()Skills()
}type WorkIF interface {Class()Income()
}type Person struct { //人的基本Name stringage  intsex  string
}/*Person实现PersonIF接口
*/
func (this *Person) Status() {println("Status")
}func (this *Person) Skills() {println("Skills")
}/*Person实现WorkIF接口*/
func (this *Person) Class() {println("Class")
}	
func (this *Person) Income() {println("Income")
}

⭐️ 调用则和Java差不多,这里可以直接使用Interfer来创建一个变量调用,然后就可以调用实现interfer内的方法

type PersonIF interface {Status()Skills()
}type WorkIF interface {Class()Income()
}type Person struct { //人的基本Name stringage  intsex  string
}/*
Person实现PersonIF接口
*/
func (this *Person) Status() {println("Status")
}func (this *Person) Skills() {println("Skills")
}/*
Person实现WorkIF接口
*/
func (this *Person) Class() {println("Class")
}
func (this *Person) Income() {println("Income")
}//Person类自己的方法
func (this *Person) GetTest() {println("GetTest")
}func main() {p1 := Person{"张三", 18, "男"} //创建一个对象pIn := PersonIF(&p1) //实现接口pIn.Status() //调用接口的方法,接口无法调用类自己的方法,也就是无法调用GetTestpIn.Skills()p1.GetTest()fmt.Println(pIn)  //接口变量包含了接受实列的值和指向对应方法表的指针,这里输出的是实列的值
}

⭐️ 接口也可以内嵌接口

type PersonIF interface {Status()Skills()WorkIF
}type WorkIF interface {Class()Income()
}
func main() {p1 := Person{"张三", 18, "男"} //创建一个对象pIn := PersonIF(&p1)        //实现接口pIn.Class()			//可以调用嵌套接口中的实现fmt.Println(pIn)
}

2️⃣ 接口类型断言和空接口

⭐️ Go语言有个万能的类型通配符,是一个空接口interfer{},因为在Go中任何接口类型都实现了空接口,类似于Java中的Object

⭐️ 每个空接口变量在内存中占据两个字长,一个是用来存储包含的类型,一个是存储数据或者指向数据的指针

func main() {Test(1)Test("abc")Test(1.111)
}func Test(i interface{}) { //空类型fmt.Println(i)
}//输出:
1
abc  
1.111
🌟 空接口实现存储不同数据类型的切片/数组

⭐️ 直接使用type给空接口一个别名类型,然后创建这个接口别名类型的切片/数组(真的太聪明了这个办法)

package mainimport "fmt"type Empty interface{}type Vector struct {e []Empty
}func (this *Vector) NewInit() { //初始化切片this.e = make([]Empty, 5)
}func (this *Vector) Set(i int, a interface{}) { //将元素插入指定索引this.e[i] = a
}
func (this *Vector) Get(i int) { //获取指定索引元素并输出fmt.Println(this.e[i])
}func main() {var v Vectorv.NewInit() //初始化v.Set(0, 22) //整数v.Set(1, "222") //stringv.Set(2, 13.14) //float32v.Get(0)v.Get(1)v.Get(2)
}//输出:
22
222
13.14
🌟 复制切片到空接口切片

⭐️ 如果你需要将切片复制到一个空接口切片中需要通过for-range,不能直接传递

func main() {dataSlice := make([]int, 0)dataSlice = append(dataSlice, 1, 2, 3, 4, 5, 6)emptySlice := make([]interface{}, 10)for i, j := range dataSlice {emptySlice[i] = j}fmt.Println(emptySlice)
}

⭐️ 一个接口的值是可以赋值给另外一个接口变量,只要底层类型实现了必要的方法

🌟 类型断言

⭐️ 如果变量是一个接口变量,则直接可以使用断言机制,直接就是接口变量.(类型)即可,注意一定要是接口变量

func main() {Test(1)Test("abc")
}func Test(i interface{}) { //空类型if value, ok := i.(string); ok {fmt.Printf("值: %v 是string\n", value)} else {fmt.Printf("值: %v 不是string\n", i)}
}

⭐️ 还有一种是type-switch

func main() {Test(1)Test("abc")
}func Test(i interface{}) { //空类型switch v := i.(type) {case string:fmt.Printf("值 %v 是string\n", v)case int:fmt.Printf("值 %v 是int\n", v)case float32:fmt.Printf("值 %v 是float32\n", v)default:fmt.Printf("我不想学习\n")}
}

反射

Go 语言反射的实现原理 | Go 语言设计与实现 (draveness.me) 可以好好看看

⭐️ Go语言的反射机制也是通过空接口来实现的,在reflect包中

⭐️ 反射提供了两个简单的函数实现一个是TypeOf(获取类型信息),一个是ValueOf(获取数据的运行时表示)

它们两个对应了两个不同的接口分别是TypeValue

golang-reflection

⭐️ TypeOf返回的是一个Type接口对象,所以说它可以使用Type方法

它接受一个any类型的值

type any = interface{} //空接口
// TypeOf函数返回给定参数i的类型。
// 
// 参数:
// i - 任意类型的值。
// 
// 返回值:
// Type - 表示参数i类型的类型值。
func TypeOf(i any) Type {// 将i转换为emptyInterface类型,以获取其动态类型信息。eface := *(*emptyInterface)(unsafe.Pointer(&i))// 使用noescape函数确保i的指针不会逃逸,这样做是为了避免i的生命周期被延长。// 这是基于Value.typ字段的注释中提到的安全理由。return toType((*abi.Type)(noescape(unsafe.Pointer(eface.typ))))
}

⭐️ 在Type接口中内定义了很多方法

type Type interface {// 基本属性Align() int           // 返回类型值对齐所需的字节数。FieldAlign() int      // 返回类型作为结构体字段时的对齐要求。// 方法相关Method(int) Method    // 获取方法集合中的第i个方法。NumMethod() int       // 返回方法总数。MethodByName(string) (Method, bool) // 通过名称查找方法。// 类型基本信息Name() string        // 返回类型名。PkgPath() string     // 返回类型所在的包路径。Size() uintptr       // 返回类型的大小。String() string      // 返回类型表示的字符串。Kind() Kind          // 返回类型的基本类别。// 类型兼容性Implements(Type) bool    // 检查是否实现某个接口。AssignableTo(Type) bool // 检查值是否可直接赋值给另一类型。ConvertibleTo(Type) bool // 检查值是否可通过类型转换匹配另一类型。Comparable() bool       // 指示类型值是否可比较。// 特定类型特有操作Elem() Type             // 对数组/切片/指针/映射/通道,返回元素类型。ChanDir() ChanDir        // 对通道类型,返回通信方向。IsVariadic() bool       // 对函数类型,检查是否可变参数。Key() Type              // 对映射类型,返回键的类型。Len() int               // 对数组类型,返回其长度。// 结构体操作Field(int) StructField  // 获取结构体的第i个字段信息。NumField() int         // 返回结构体字段数量。FieldByIndex([]int) StructField // 通过嵌套索引获取字段信息。FieldByName(string) (StructField, bool) // 通过字段名获取字段信息。FieldByNameFunc(func(string) bool) (StructField, bool) // 通过匹配函数查找字段。// 函数类型操作In(int) Type           // 获取函数的第i个输入参数类型。NumIn() int            // 返回函数输入参数数量。Out(int) Type          // 获取函数的第i个输出参数类型。NumOut() int           // 返回函数输出参数数量。// 内部方法,用户通常无需直接调用common(), uncommon() *internal // 返回底层实现相关的数据结构。
}

⭐️ ValueOf 放回一个Value对象,接受的也是一个任意类型的值

// ValueOf 是一个将任意类型转换为 Value 类型的函数。
// 参数 i 为任意类型的值,表示需要转换的值。
// 返回值为 Value 类型,表示转换后的值。
// 如果输入值 i 为 nil,则返回一个空的 Value。
func ValueOf(i any) Value {// 检查输入值是否为 nil,如果是则返回空的 Valueif i == nil {return Value{}}// 如果满足特定条件(go121noForceValueEscape 为 false),则标记 i 为逃逸变量if !go121noForceValueEscape {escapes(i)}// 将输入值 i 解包为 Value 类型并返回return unpackEface(i)
}

⭐️ 在Value中也定义了一些方法,反射包中 reflect.Value 的类型与 reflect.Type 不同,它被声明成了结构体。这个结构体没有对外暴露的字段,但是提供了获取或者写入数据的方法

type Value struct {// 内部结构未公开,实际包含值和类型等信息
}// 实例化Value的方法通常通过reflect.ValueOf或reflect零值的间接操作获得// Kind 返回此Value所持有的值的类型种类。
func (v Value) Kind() Kind// Type 返回此Value所持有的值的具体类型信息。
func (v Value) Type() Type// Bool 获取bool类型的值,如果Value不是bool类型则会panic。
func (v Value) Bool() bool// Int 获取整数值,类型必须是整数类型,否则会panic。
func (v Value) Int() int64// Float 获取浮点数值,类型必须是浮点数类型,否则会panic。
func (v Value) Float() float64// String 获取字符串值,类型必须是string,否则会panic。
func (v Value) String() string// Interface 转换Value为interface{}类型,几乎所有类型都可以通过此方法获取。
func (v Value) Interface() interface{}// Set 设置Value的值,新值必须是可设置的并且类型匹配。
func (v Value) Set(x Value) // SetBool 设置bool类型的值,Value必须是可设置的bool类型。
func (v Value) SetBool(x bool)// SetInt 设置整数值,Value必须是可设置的整数类型。
func (v Value) SetInt(x int64)// SetFloat 设置浮点数值,Value必须是可设置的浮点数类型。
func (v Value) SetFloat(x float64)// SetString 设置字符串值,Value必须是可设置的string类型。
func (v Value) SetString(x string)// Call 对于函数或方法Value,执行调用并返回结果。
func (v Value) Call(in []Value) []Value// Elem 如果Value是一个指针,返回它指向的值的Value;否则返回自身。
func (v Value) Elem() Value// Field 获取结构体Value的第i个字段的Value。
func (v Value) Field(i int) Value// FieldByName 获取结构体Value的名为name的字段的Value。
func (v Value) FieldByName(name string) Value// Index 对于数组、slice或map的Value,返回索引i处的元素Value。
func (v Value) Index(i int) Value// MapIndex 对于map的Value,返回key对应的元素Value。
func (v Value) MapIndex(key Value) Value// CanSet 返回此Value是否可被设置,即是否可以更改其底层值。
func (v Value) CanSet() bool// IsZero 判断Value的底层值是否为零值。
func (v Value) IsZero() bool

⭐️ Type方法实列

type Person struct {Name stringAge  intSex  string
}func (this *Person) toString() string {return fmt.Sprintf("Nmae: %v,Age: %v,Sex: %v", this.Name, this.Age, this.Sex)
}func main() {p1 := Person{"张三", 18, "男"}p1_demo := reflect.TypeOf(p1)             //获取类型信息fmt.Println(p1_demo.Name())               //输出类名fmt.Println(p1_demo.Size())               //输出类型大小for i := 0; i < p1_demo.NumField(); i++ { //放回结构体字段的个数,然后输出遍历field := p1_demo.Field(i) //获取字段fmt.Println(field.Name, field.Type, field.Offset)}for i := 0; i < p1_demo.NumMethod(); i++ { //放回结构体方法的个数,然后输出遍历method := p1_demo.Method(i) //获取方法fmt.Println(method.Name, method.Type)}//数组p2 := [1]Person{Person{"张三", 18, "男"}}p2_demo := reflect.TypeOf(p2)fmt.Println(p2_demo.Len()) //输出数组长度}

⭐️ Value方法实现,可以使用ValueOf方法来修改值

package mainimport ("fmt""reflect"
)type Person struct {Name stringAge  intSex  string
}func main() {// 使用指针来确保可以通过反射修改值p1 := Person{"张三", 18, "男"}fmt.Println(p1)// 获取p1的反射值p1_demo := reflect.ValueOf(&p1)// 由于p1是指针,我们先获取其指向的值的反射值p1_val := p1_demo.Elem()// 检查是否可以设置值if p1_val.CanSet() {// 创建一个新的Person值并通过反射设置newPerson := Person{"李四", 20, "女"}p1_val.Set(reflect.ValueOf(newPerson))} else {fmt.Println("无法设置值")}// 打印修改后的值fmt.Printf("%#v\n", p1)     // 使用%#v格式化输出结构体细节fmt.Println(p1_demo.Kind()) // 输出类型信息
}

欢迎关注我,继续探讨技术,如果觉得写的不错动动小手点个小赞,如果觉得我还有哪些不足可以私信哦

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

QT6学习第六天 初识QML

QT6学习第六天 创建Qt Quick UI项目使用Qt Quick DesignerQML 语法基础导入语句 import对象 object 和属性 property布局注释表达式和属性绑定QML 编码约定 设置应用程序图标 创建Qt Quick UI项目 如果你有只测试QML相关内容快速显示界面的需求&#xff0c;这时可以创建Qt Qui…

图解RabbitMQ七种工作模式生产者消费者模型的补充

文章目录 1.消费者模型2.生产者-消费者模型注意事项2.1资源释放顺序问题2.2消费者的声明问题2.3虚拟机和用户的权限问题 3.七种工作模式3.1简单模式3.2工作模式3.3发布/订阅模式3.4路由模式3.5通配符模式3.6RPC通信3.7发布确认 1.消费者模型 之前学习的这个消息队列的快速上手…

C-操作符

操作符种类 在C语言中&#xff0c;操作符有以下几种&#xff1a; 算术操作符 移位操作符 位操作符 逻辑操作符 条件操作符 逗号表达式 下标引用&#xff0c;函数调用 拓展&#xff1a;整型提升 我们介绍常用的几个 算术操作符 &#xff08;加&#xff09;&#xff…

使用 Spring Boot 和 GraalVM 的原生镜像

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家&#xff0c;历代文学网&#xff08;PC端可以访问&#xff1a;历代文学&#xff0c;移动端可微信小程序搜索“历代文学”&#xff09;总架构师&#xff0c;15年工作经验&#xff0c;精通Java编程&#xff0c;高并发设计&#xf…

基于Java Springboot宠物医院微信小程序

一、作品包含 源码数据库设计文档万字PPT全套环境和工具资源部署教程 二、项目技术 前端技术&#xff1a;Html、Css、Js、Vue、Element-ui 数据库&#xff1a;MySQL 后端技术&#xff1a;Java、Spring Boot、MyBatis 三、运行环境 开发工具&#xff1a;IDEA/eclipse 微信…

Tree搜索二叉树、map和set_数据结构

数据结构专栏 如烟花般绚烂却又稍纵即逝的个人主页 本章讲述数据结构中搜索二叉树与HashMap的学习&#xff0c;感谢大家的支持&#xff01;欢迎大家踊跃评论&#xff0c;感谢大佬们的支持! 目录 搜索二叉树的概念二叉树搜索模拟实现搜索二叉树查找搜索二叉树插入搜索二叉树删除…

C#使用ExcelDataReader读取Xlsx文件为DataTable对象

创建控制台项目 在NuGet中安装ExcelDataReader.DataSet 3.7.0 创建一个xlsx文件 测试代码 读取xlsx文件内容&#xff0c;为一个DataTable对象。 读取xlsx时&#xff0c;xlsx文件不能被其他软件打开&#xff0c;否则会报“进程无法访问此文件”的错。 using ExcelDataRead…

【JavaEE初阶】应是天仙狂醉,乱把白云揉碎 - (重点)线程

本篇博客给大家带来的是线程的知识点, 由于内容较多分几天来写. &#x1f40e;文章专栏: JavaEE初阶 &#x1f680;若有问题 评论区见 ⭐欢迎大家点赞 评论 收藏 分享 ❤❤❤ 如果你不知道分享给谁,那就分享给薯条. 你们的支持是我不断创作的动力 . 1. 认识线程 1.1 概念 )1 …

精准用户获取与私域流量运营:多商户链动 2+1 模式商城小程序的赋能策略

摘要&#xff1a;本文聚焦于精准用户对商业运营的核心价值&#xff0c;深入剖析获取精准用户的有效途径&#xff0c;特别围绕目标用户画像及出没场景展开分析。同时&#xff0c;探讨在私域流量构建进程中&#xff0c;多商户链动 21 模式商城小程序如何融入精准用户运营体系&…

Spring Boot教程之十一:获取Request 请求 和 Put请求

如何在 Spring Boot 中获取Request Body&#xff1f; Java 语言是所有编程语言中最流行的语言之一。使用 Java 编程语言有几个优点&#xff0c;无论是出于安全目的还是构建大型分发项目。使用 Java 的优点之一是 Java 试图借助类、继承、多态等概念将语言中的每个概念与现实世…

DVWA靶场文件包含(File Inclusion)通关教程(high级别)

目录 DVWA 靶场建立闯关 DVWA 靶场建立 需要的东西&#xff1a; phpStudy&#xff1a; 链接&#xff1a; phpStudy 提取码&#xff1a;0278 DVWA-master 链接&#xff1a; DVWA靶场 提取码&#xff1a;0278 建议在虚拟机中操作&#xff0c;以防数据库冲突&#xff0c;下面有…

基于yolov8、yolov5的铝材缺陷检测识别系统(含UI界面、训练好的模型、Python代码、数据集)

摘要&#xff1a;铝材缺陷检测在现代工业生产和质量管理中具有重要意义&#xff0c;不仅能帮助企业实时监控铝材质量&#xff0c;还为智能化生产系统提供了可靠的数据支撑。本文介绍了一款基于YOLOv8、YOLOv5等深度学习框架的铝材缺陷检测模型&#xff0c;该模型使用了大量包含…

力扣刷题TOP101:8.BM10 两个链表的第一个公共结点

目录&#xff1a; 目的 思路 复杂度 记忆秘诀 python代码 目的 两个无环的单向链表&#xff0c;它们的第一个公共结点{{6,7}。 思路 这个任务是找到两个链表的第一个公共结点。可以看作两个心机boy偷偷补课翻车事件。平时嘴上说自己在家玩游戏&#xff0c;实际上背地里都偷…

哪些行业对六西格玛管理方法的需求较大?

六西格玛作为一种追求极致质量和流程优化的管理哲学&#xff0c;自诞生以来&#xff0c;便在多个行业中展现出了巨大的应用价值。该方法通过定义、测量、分析、改进和控制&#xff08;DMAIC&#xff09;五个阶段&#xff0c;帮助企业实现流程的持续改进&#xff0c;提高产品质量…

Spring Web MVC其他扩展(详解下)

文章目录 Spring MVC其他扩展&#xff08;下&#xff09;异常处理异常处理机制声明式异常好处基于注解异常声明异常处理 拦截器拦截器概念拦截器使用拦截器作用位置图解拦截器案例拦截器工作原理源码 参数校验校验概述操作演示SpringMVC自定义参数验证ValueObject(VO) 文件上传…

排序学习整理(1)

1.排序的概念及运用 1.1概念 排序&#xff1a;所谓排序&#xff0c;就是使⼀串记录&#xff0c;按照其中的某个或某些关键字的大小&#xff0c;递增或递减的排列起来的操作&#xff0c;以便更容易查找、组织或分析数据。 1.2运用 购物筛选排序 院校排名 1.3常见排序算法 2.实…

【linux学习指南】Linux进程信号产生(三) 硬件异常除零出错?野指针异常?core文件

文章目录 &#x1f4dd;前言&#x1f320;模拟除0&#x1f309;除0出错&#xff1f;&#x1f309;野指针异常? &#x1f320;⼦进程退出coredump&#x1f309;Core Dump &#x1f6a9;总结 &#x1f4dd;前言 硬件异常被硬件以某种⽅式被硬件检测到并通知内核,然后内核向当前…

【人工智能-科普】图神经网络(GNN):与传统神经网络的区别与优势

文章目录 图神经网络(GNN):与传统神经网络的区别与优势什么是图神经网络?图的基本概念GNN的工作原理GNN与传统神经网络的不同1. 数据结构的不同2. 信息传递方式的不同3. 模型的可扩展性4. 局部与全局信息的结合GNN的应用领域总结图神经网络(GNN):与传统神经网络的区别与…

青藤云安全携手财信证券,入选金融科技创新应用优秀案例

11月29日&#xff0c;由中国信息通信研究院主办的第四届“金信通”金融科技创新应用案例评选结果正式发布。财信证券与青藤云安全联合提交的“基于RASP技术的API及数据链路安全治理项目”以其卓越的创新性和先进性&#xff0c;成功入选金融科技创新应用优秀案例。 据悉&#x…

Python系列 - MQTT协议

Python系列 - MQTT协议 资源连接 MQTT的介绍和应用场景的示例说明 一、什么是MQTT 百度关于MQTT的介绍如下&#xff1a; MQTT(消息队列遥测传输)是ISO 标准(ISO/IEC PRF 20922)下基于发布订阅范式的消息协议。它工作在 TCP/IP协议之上&#xff0c;是为硬件性能低下的远程设…