当前位置: 首页 > news >正文

Go 实现多态和 参数的动态个数及动态类型

引子

go语言作为静态(编译期类型检测)强类型(手写代码进行类型转换)语言, 要想实现 动态语言的鸭子类型的调用方法,做到 一个入参是不同类型,还是有些麻烦的;

需求

  • 希望写代码时像python一样的鸭子类型,不用管参数类型,都可以调用同一个方法;
  • 希望 入参像python一样 能够在 个数上动态变化及类型上也动态变化;

go语言实现如上需求需要的技术

interface实现

  • interface作为 定义一个类型的接口 实现 多态,只要 具体的结构体实现了 此接口中定义的方法,编译器就会进行隐式的类型转换,从而实现多态;
  • interface作为 函数参数 实现 动态的参数类型;

多态定义

  • 面向对象3大特性之一
  • 指 一类事务有多种形态,如 动物类 具体有 猫,狗,猪等, 他们都会"走路", 当我们调用 “走路” 方法时,不用考虑具体是什么动物,只用调用即可;

代码实现

// Package main
// @Description: 所谓interface(接口)  类似python中的 多个对象 都有一个相同的方法;  将 多个对象根据条件 灵活的复制给一个变量,此变量可以调用这个方法;
//
//	接口 是一种动态类型, 才可以实现 多个结构体 赋值给一个变量;
//	接口 里实现的相关方法,可以是指针接收者实现 或者 值接收者实现;不同点在于 在接口中 值接收者 可以接收值和指针2种方式;而 指针接收者 只能接收 指针类型; 所有写代码时 最好写 值接收者;
//	接口就像一个协议,要想调用此接口的 结构体必须实现 接口里要求的方法才行;
package mainimport ("errors""fmt""math/rand""reflect"
)// FinanceCal
// @Description: 定义一个接口, 里面实现了2个方法; 要想调用此接口,则 调用方 必须实现接口里的2个方法
type FinanceCal interface {//// QuerySQL//  @Description: 动态类型的动态入参个数的 方法(基本就是动态语言的特点了)//  @param ...interface{}//  @return string//QuerySQL(...interface{}) string//// buy//  @Description:一个普通未带参数的方法//buy()////  sell//  @Description: 带参数的 方法(静态语言的一般方法)//  @param amount:卖出金额//sell(amount int)
}// FinanceQuerySql
//
//	@Description: 鸭子类型的多态,只要实现了此接口下的所有方法的结构体 都可以调用此方法
//	@param obj:
//	@param params:
//	@return string:
func FinanceQuerySql(obj FinanceCal, params ...interface{}) string {return obj.QuerySQL(params...)
}// Fund
// @Description: 基金结构体
type Fund struct {////  name//  @Description://name string
}// QuerySQL
//
//	@Description:生成查询sql
//	@receiver f:
//	@param params: 动态入参,入参个数为1个或0个;参数类型 是int类型
//	@return string:
func (f Fund) QuerySQL(params ...interface{}) string {//参数处理部分//参数长度校验//第一个参数处理l := len(params)if l > 1 {panic(errors.New("入参个数错误,应该<=1个参数"))}var num interface{}if l == 1 {paramNum := params[0]t := reflect.ValueOf(paramNum)if t.Kind() != reflect.Int {panic(errors.New("数据类型错误,应该是Int类型"))}num = paramNum.(int)}//业务处理部分sql := fmt.Sprintf("select * from fund where name in \"%v\"", f.name)if num != nil {sql += fmt.Sprintf(" and num=%v", num)}fmt.Println("fund sql查询语句为:", sql)return sql
}// buy
//
//	@Description: 购买基金
//	@receiver f:
func (f Fund) buy() {fmt.Printf("基金 %s 购买\n", f.name)
}// sell
//
//  @Description:
//  @receiver f:
//  @param amount:
func (f Fund) sell(amount int) {fmt.Printf("基金 %s 卖出 %d 元\n", f.name, amount)
}// Stock
// @Description: 股票结构体
type Stock struct {////  name//  @Description://name string
}// QuerySQL
//
//	@Description:生成查询sql
//	@receiver f:
//	@param params: 动态入参,入参个数为 0~2个;第一个参数 是int类型;第二个参数是 string类型;
//	@return string:
func (f Stock) QuerySQL(params ...interface{}) string {//参数处理部分//参数长度校验l := len(params)if l > 2 {panic(errors.New("入参个数错误,应该<=2个参数"))}//第一个参数处理var num interface{}if l >= 1 {paramNum := params[0]t := reflect.ValueOf(paramNum)if t.Kind() != reflect.Int {panic(errors.New("第一个参数 数据类型错误,应该是Int类型"))}num = paramNum.(int)}var manager interface{}if l == 2 {paramString := params[1]t := reflect.ValueOf(paramString)if t.Kind() != reflect.String {panic(errors.New("第二个参数 数据类型错误,应该是String类型"))}manager = paramString.(string)}//业务处理部分sql := fmt.Sprintf("select * from stock where name in \"%v\"", f.name)if num != nil {sql += fmt.Sprintf(" and num=%v", num)}if manager != nil {sql += fmt.Sprintf(" and manager=\"%v\"", manager)}fmt.Println("stock sql查询语句为:", sql)return sql
}// buy
//
//	@Description: 购买股票
//	@receiver f:
func (f Stock) buy() {fmt.Printf("股票 %s 购买\n", f.name)
}// sell
//
//  @Description:
//  @receiver f:
//  @param amount:
func (f Stock) sell(amount int) {fmt.Printf("股票 %s 卖出 %d 元\n", f.name, amount)
}func main() {//此方法一般是判断 结构体是否实现了 接口的方法,没实现 则直接编译时报错var _ FinanceCal = Fund{}var _ FinanceCal = (*Stock)(nil)//  定义 结构体 的变量,并实例化fund := Fund{"00001.OF"}//  定义一个 接口类型的变量 如下2种方法//var f = FinanceCal(fund)var fc FinanceCal = fund//  将 结构体 变量赋值给 接口类型变量//  fund的值赋值给fcfc = fund//  接口类型变量 调用对应的方法fc.buy()fc.QuerySQL(1)fc.QuerySQL()fc.sell(100)//  stock的指针赋值给fcstock := Stock{"023123.SZ"}fc = stock//  接口类型变量 调用对应的方法, 也就是 类似python中,无论 哪个对象赋值给此变量,都可以访问 这些对象有的某个方法fc.QuerySQL()fc.QuerySQL(1)fc.QuerySQL(1, "经理")fc.buy()fc.sell(1000)// 鸭子类型的实现var obj FinanceCalvar params []interface{}if rand.Intn(10)%2 == 0 {obj = fundparams = []interface{}{666}fmt.Println("对象是Fund结构体的示例")} else {obj = stockparams = []interface{}{345, "经理A"}fmt.Println("对象是Stock结构体的示例")}FinanceQuerySql(obj, params...)
}

执行结果
在这里插入图片描述

总结

  • go语言中 函数参数或变量类型里的interface(如 : func add(input interface{})) 实现了 不同类型的值 赋值的功能,实现了动态类型语言的灵活性;
  • go语言中 对变量初始化为interface{}类型,实现了 业务代码中经常需要判断 参数是否有值的功能,因为 go语言定义为具体类型,都会有默认值,如 int类型默认值为0,而实际业务中0也可能是有效数据,无法作为是否有值的判断;
// go代码 判断值是否为空
var num =interface{}
if num==nil{print("num没有值")
}
num=1
print("num被赋值为int类型的值为1")
# python代码判断值是否为空
num=None
if num==None:print("num没有值")
  • go语言中的interface作为 定义一个类型的接口,只要实现了其中定义的所有方法,那么就实现了这个类型的接口;以此为基础 实现 多态功能,此功能 较为实用及重要;

相关文章:

Go 实现多态和 参数的动态个数及动态类型

引子 go语言作为静态(编译期类型检测)强类型(手写代码进行类型转换)语言, 要想实现 动态语言的鸭子类型的调用方法,做到 一个入参是不同类型,还是有些麻烦的; 需求 希望写代码时像python一样的鸭子类型,不用管参数类型,都可以调用同一个方法;希望 入参像python一样 能够在 个…...

vue 指令

Vue 提供了很多指令&#xff0c;如&#xff1a;v-model, v-show&#xff0c;v-if等等&#xff0c;有利于应付开发时出现的各种情况。Vue 也提供了自定义指令&#xff0c;有利于开发者将某些通用性功能封装成一个指令&#xff0c;进行全局或局部注册。如&#xff1a;复制指令&am…...

APP违法违规收集使用个人信息合规评流程和范围

近期&#xff0c;工信部通报2023年第1批《侵害用户权益行为的APP通报》&#xff08;总第27批&#xff09;&#xff0c;共通报46款APP&#xff08;SDK&#xff09;&#xff0c;这些被责令限期整改的APP&#xff08;SDK&#xff09;&#xff0c;涉及的问题主要包括3个方面&#x…...

【力扣2379】 得到 K 个黑块的最少涂色次数(c++100%)

给你一个长度为 n 下标从 0 开始的字符串 blocks &#xff0c;blocks[i] 要么是 W 要么是 B &#xff0c;表示第 i 块的颜色。字符 W 和 B 分别表示白色和黑色。给你一个整数 k &#xff0c;表示想要 连续 黑色块的数目。每一次操作中&#xff0c;你可以选择一个白色块将它 涂成…...

[2.2.2]进程调度的时机、方式、切换与过程

文章目录第二章 进程管理进程调度的时机、方式、切换与过程&#xff08;一&#xff09;进程调度的时机&#xff08;二&#xff09;进程调度的方式&#xff08;三&#xff09;进程的切换与过程小结第二章 进程管理 进程调度的时机、方式、切换与过程 时机 什么时候需要进程调度…...

第24篇:Java包装类知识深度分析

目录 1、包装类背景 2、包装类的优点 3、包装类与基本类型关系 4、代码示例...

常见问题整理1

目录 偏差和方差 欠拟合underfitting 过拟合overfitting 梯度消失和梯度爆炸 归一化 偏差和方差 偏差&#xff1a;算法期望预测和真实预测之间的偏差程度。反应的是模型本身的拟合能力。 方差&#xff1a;度量了同等大小的训练集的变动导致学习性能的变化&#xff0c;刻画…...

体验Linux 块设备驱动实验(模拟块)

目录 一、块设备 二、块设备驱动框架 1、块设备的注册和注销 2、gendisk 结构体 3、block_device_operations 结构体 4、块设备 I/O 请求过程 ①、请求队列 request_queue ②、bio 结构 三、编写驱动之请求队列 1、修改makefile 2、基本的驱动框架​编辑 3、添加头文…...

一文搞懂Linux时区设置、自定义时区文件

概念介绍 常说的 Linux 系统时钟有两个 一个是硬件时钟&#xff08;RTC&#xff09;&#xff0c;即BIOS时间&#xff0c;一般保存的是 GMT0 时间&#xff0c;没时区、夏令时的概念 一个是当地时钟&#xff08;LTC&#xff09;&#xff0c;即我们日常经常看到的时间&#xff0…...

Java实例实验项目大全源码企业通讯打印系统计划酒店图书学生管理进销存商城门户网站五子棋

wx供重浩&#xff1a;创享日记 对话框发送&#xff1a;java实例 获取完整源码源文件视频讲解文档资料等 文章目录1、企业通讯2、快递打印系统3、开发计划管理系统4、酒店管理系统5、图书馆管理系统6、学生成绩管理系统7、进销存管理系统8、神奇Book——图书商城9、企业门户网站…...

基于nvidia xavier智能车辆自动驾驶域控制器设计与实现-百度Apollo架构(二)

智能车辆操作系统 智能车辆操作系统是智能车辆系统的重要组成部分。现代汽车软件组件通常首 先由不同的供应商开发&#xff0c;然后在有限的资源下由制造商进行集成[42]。智能车辆操作 系统需要采用模块化和分层化设计思想来兼容传感器、分布式通信和自动驾驶通用 框架等模块&a…...

考研408 王道计算机考研 (初试/复试) 网课笔记总结

计算机初试、复试笔记总结&#xff08;导航栏&#xff09;&#x1f4dd; 一、初试 408 408 - 1. 数据结构与算法 数据结构与算法 笔记导航&#x1f6a5;&#x1f6a5;&#x1f6a5; &#x1f96c; 第一章 绪论(无)&#x1f955; 第二章 线性表&#x1f96a; 第三章 栈和队列&…...

[Java·算法·中等]LeetCode34. 在排序数组中查找元素的第一个和最后一个位置

每天一题&#xff0c;防止痴呆题目示例分析思路1题解1&#x1f449;️ 力扣原文 题目 给你一个按照非递减顺序排列的整数数组 nums&#xff0c;和一个目标值 target。请你找出给定目标值在数组中的开始位置和结束位置。 如果数组中不存在目标值 target&#xff0c;返回 [-1,…...

SAP BTEs的简介及实现

一、认识BTE BTE&#xff08;Business Transaction Event&#xff09;也称之为“业务交易事件”,一般的增强(Tcode:SMOD|CMOD)依旧使用ABAP进行二次开发,然而BTE则提供了RFC调用其它产品的可能(Tcode:FIBF)。BTE的设计思路更加简单&#xff0c;和BADI有点类似。在标准程序中留有…...

如何利用海外主机服务提高网站速度?

网站速度是任何在线业务成功的关键。快速的网站速度可以让用户更快地访问您的网站&#xff0c;增加页面浏览量。对于拥有全球用户的网站而言&#xff0c;选择一个海外主机服务商是提高网站速度的有效方法之一。下面是一些利用海外主机服务(如美国主机、香港主机)提高网站速度的…...

【SpringMVC】 一文掌握 》》》 @RequestMapping注解

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ RequestMapping注解一、SpringMVC环境准备1.相…...

高三应该怎么复习

高三是学生们备战高考的重要一年&#xff0c;正确有序的复习可以有效地提高复习效率&#xff0c;下面是一些高效复习的方法和建议&#xff1a;1. 制定合理的学习计划和目标高三的学生要制定合理的学习计划和目标&#xff0c;适当的计划和目标可以使学习更有针对性和效率。建议根…...

如何通过C++ 将数据写入 Excel 工作表

直观的界面、出色的计算功能和图表工具&#xff0c;使Excel成为了最流行的个人计算机数据处理软件。在独立的数据包含的信息量太少&#xff0c;而过多的数据又难以理清头绪时&#xff0c;制作成表格是数据管理的最有效手段之一。这样不仅可以方便整理数据&#xff0c;还可以方便…...

Kalman Filter in SLAM (6) ——Error-state Kalman Filter (EsKF, 误差状态卡尔曼滤波)

文章目录0.前言1. IMU的误差状态空间方程2. 误差状态观测方程3. 误差状态卡尔曼滤波4. 误差状态卡尔曼滤波方程细节问题0.前言 这里先说一句&#xff1a;什么误差状态卡尔曼&#xff1f;完全就是在扯淡&#xff01; 回想上面我们推导的IMU的误差状态空间方程&#xff0c;其实…...

centos7部署KVM虚拟化

目录 centos7部署KVM虚拟化平台 1、新建一台虚拟机 2、系统内的操作 1、修改主机名 2、挂载镜像光盘 3、ssh优化 4、设置本地yum仓库 5、关闭防火墙&#xff0c;selinux 3、安装KVM 4、设置KVM网络 5、KVM部署与管理 6、使用虚拟系统管理器管理虚拟机 创建存储池 …...

【磁盘】每天掌握一个Linux命令 - iostat

目录 【磁盘】每天掌握一个Linux命令 - iostat工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景 注意事项 【磁盘】每天掌握一个Linux命令 - iostat 工具概述 iostat&#xff08;I/O Statistics&#xff09;是Linux系统下用于监视系统输入输出设备和CPU使…...

DBAPI如何优雅的获取单条数据

API如何优雅的获取单条数据 案例一 对于查询类API&#xff0c;查询的是单条数据&#xff0c;比如根据主键ID查询用户信息&#xff0c;sql如下&#xff1a; select id, name, age from user where id #{id}API默认返回的数据格式是多条的&#xff0c;如下&#xff1a; {&qu…...

leetcodeSQL解题:3564. 季节性销售分析

leetcodeSQL解题&#xff1a;3564. 季节性销售分析 题目&#xff1a; 表&#xff1a;sales ---------------------- | Column Name | Type | ---------------------- | sale_id | int | | product_id | int | | sale_date | date | | quantity | int | | price | decimal | -…...

Android 之 kotlin 语言学习笔记三(Kotlin-Java 互操作)

参考官方文档&#xff1a;https://developer.android.google.cn/kotlin/interop?hlzh-cn 一、Java&#xff08;供 Kotlin 使用&#xff09; 1、不得使用硬关键字 不要使用 Kotlin 的任何硬关键字作为方法的名称 或字段。允许使用 Kotlin 的软关键字、修饰符关键字和特殊标识…...

IP如何挑?2025年海外专线IP如何购买?

你花了时间和预算买了IP&#xff0c;结果IP质量不佳&#xff0c;项目效率低下不说&#xff0c;还可能带来莫名的网络问题&#xff0c;是不是太闹心了&#xff1f;尤其是在面对海外专线IP时&#xff0c;到底怎么才能买到适合自己的呢&#xff1f;所以&#xff0c;挑IP绝对是个技…...

RSS 2025|从说明书学习复杂机器人操作任务:NUS邵林团队提出全新机器人装配技能学习框架Manual2Skill

视觉语言模型&#xff08;Vision-Language Models, VLMs&#xff09;&#xff0c;为真实环境中的机器人操作任务提供了极具潜力的解决方案。 尽管 VLMs 取得了显著进展&#xff0c;机器人仍难以胜任复杂的长时程任务&#xff08;如家具装配&#xff09;&#xff0c;主要受限于人…...

打手机检测算法AI智能分析网关V4守护公共/工业/医疗等多场景安全应用

一、方案背景​ 在现代生产与生活场景中&#xff0c;如工厂高危作业区、医院手术室、公共场景等&#xff0c;人员违规打手机的行为潜藏着巨大风险。传统依靠人工巡查的监管方式&#xff0c;存在效率低、覆盖面不足、判断主观性强等问题&#xff0c;难以满足对人员打手机行为精…...

Golang——9、反射和文件操作

反射和文件操作 1、反射1.1、reflect.TypeOf()获取任意值的类型对象1.2、reflect.ValueOf()1.3、结构体反射 2、文件操作2.1、os.Open()打开文件2.2、方式一&#xff1a;使用Read()读取文件2.3、方式二&#xff1a;bufio读取文件2.4、方式三&#xff1a;os.ReadFile读取2.5、写…...

通过MicroSip配置自己的freeswitch服务器进行调试记录

之前用docker安装的freeswitch的&#xff0c;启动是正常的&#xff0c; 但用下面的Microsip连接不上 主要原因有可能一下几个 1、通过下面命令可以看 [rootlocalhost default]# docker exec -it freeswitch fs_cli -x "sofia status profile internal"Name …...

springboot 日志类切面,接口成功记录日志,失败不记录

springboot 日志类切面&#xff0c;接口成功记录日志&#xff0c;失败不记录 自定义一个注解方法 import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target;/***…...