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

news2024/9/28 17:27:06

引子

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

需求

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

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

interface实现

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

多态定义

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

代码实现

// Package main
// @Description: 所谓interface(接口)  类似python中的 多个对象 都有一个相同的方法;  将 多个对象根据条件 灵活的复制给一个变量,此变量可以调用这个方法;
//
//	接口 是一种动态类型, 才可以实现 多个结构体 赋值给一个变量;
//	接口 里实现的相关方法,可以是指针接收者实现 或者 值接收者实现;不同点在于 在接口中 值接收者 可以接收值和指针2种方式;而 指针接收者 只能接收 指针类型; 所有写代码时 最好写 值接收者;
//	接口就像一个协议,要想调用此接口的 结构体必须实现 接口里要求的方法才行;
package main

import (
	"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的值赋值给fc
	fc = fund
	//  接口类型变量 调用对应的方法
	fc.buy()
	fc.QuerySQL(1)
	fc.QuerySQL()
	fc.sell(100)

	//  stock的指针赋值给fc
	stock := Stock{"023123.SZ"}
	fc = stock
	//  接口类型变量 调用对应的方法, 也就是 类似python中,无论 哪个对象赋值给此变量,都可以访问 这些对象有的某个方法
	fc.QuerySQL()
	fc.QuerySQL(1)
	fc.QuerySQL(1, "经理")
	fc.buy()
	fc.sell(1000)

	// 鸭子类型的实现
	var obj FinanceCal
	var params []interface{}
	if rand.Intn(10)%2 == 0 {
		obj = fund
		params = []interface{}{666}
		fmt.Println("对象是Fund结构体的示例")
	} else {
		obj = stock
		params = []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作为 定义一个类型的接口,只要实现了其中定义的所有方法,那么就实现了这个类型的接口;以此为基础 实现 多态功能,此功能 较为实用及重要;

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

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

相关文章

Kalman Filter in SLAM (1) ——Data Fusion and Kalman Filter(数据融合和卡尔曼滤波)

文章目录0. 参考资料1. Intro Example 例子引入1.1. 测量硬币直径1.2. 思考2. Data Fusion 数据融合2.1. 数据融合在做什么&#xff1f;2.2. 数据融合的前提——不确定度2.3. 数据融合的结果——统计意义下的最优估计3. State Space Representation 状态空间表达式3.1. 状态方程…

大数据 | (二)SSH连接报错Permission denied

大数据 | &#xff08;三&#xff09;centos7图形界面无法执行yum命令&#xff1a;centos7图形界面无法执行yum命令 哈喽&#xff01;各位CSDN的朋友们大家好&#xff01; 今天在执行Hadoop伪分布式安装时&#xff0c;遇到了一个问题&#xff0c;在此跟大家分享&#xff0c; …

ThreadLocal的内部结构和源码探究

目录一. ThreadLocal的内部结构1 常见的误解2 现在的设计3 这样设计的好处二. ThreadLocal的核心方法源码1 set方法2 get方法3 remove方法**4 initialValue方法**三. ThreadLocalMap源码分析1 基本结构2 弱引用和内存泄漏3 hash冲突的解决一. ThreadLocal的内部结构 ​ 通过之…

【C++知识点】STL 容器总结

✍个人博客&#xff1a;https://blog.csdn.net/Newin2020?spm1011.2415.3001.5343 &#x1f4da;专栏地址&#xff1a;C/C知识点 &#x1f4e3;专栏定位&#xff1a;整理一下 C 相关的知识点&#xff0c;供大家学习参考~ ❤️如果有收获的话&#xff0c;欢迎点赞&#x1f44d;…

2月榜单丨飞瓜数据B站UP主排行榜(哔哩哔哩平台)发布!

飞瓜轻数发布2023年2月飞瓜数据UP主排行榜&#xff08;B站平台&#xff09;&#xff0c;通过充电数、涨粉数、成长指数三个维度来体现UP主账号成长的情况&#xff0c;为用户提供B站号综合价值的数据参考&#xff0c;根据UP主成长情况用户能够快速找到运营能力强的B站UP主。飞瓜…

(蓝桥真题)剪格子(搜索+剪枝)

样例1输入&#xff1a; 3 3 10 1 52 20 30 1 1 2 3 样例1输出&#xff1a; 3 样例2输入&#xff1a; 4 3 1 1 1 1 1 30 80 2 1 1 1 100 样例2输出&#xff1a; 10 分析&#xff1a;这道题目我们直接从(1,1)点开始进行dfs搜索即可&#xff0c;但是需要注意一点的是我们搜…

FPGA和IC设计怎么选?哪个发展更好?

很多人纠结FPGA和IC设计怎么选&#xff0c;其实往小了说&#xff0c;要看你选择的具体是哪个方向岗位。往大了说&#xff0c;将来你要是走更远&#xff0c;要成为大佬&#xff0c;那基本各个方向的都要有涉及的。 不同方向就有不同的发展&#xff0c;目前在薪资上IC设计要比FP…

Vue3返回顶部组件及返回顶部js封装

介绍 vue3中,封装监听页面滚动的js, 及页面滚动到一定像素时,显示返回顶部的按钮,点击按钮会有放大的动画,并逐渐滚动到顶部的组件。效果如下: 代码 封装js,监听屏幕滚动事件,以及是否显示返回顶部的按钮; 在项目目录下新建 utils文件夹,并在该文件夹下创建index.…

国外SEO优化的重要性及应对策略

SEO是指搜索引擎优化&#xff0c;是一种通过优化网站的结构和内容&#xff0c;提高网站在搜索引擎中的排名&#xff0c;从而吸引更多的流量和潜在客户的过程。 国外SEO优化尤为重要&#xff0c;因为搜索引擎在全球范围内广泛使用&#xff0c;而谷歌是全球最受欢迎的搜索引擎之…

java Math类 和 System类 详解(通俗易懂)

Math类介绍Math类常用方法及演示System类简介System类常用方法及演示一、前言本节内容是我们《API-常用类》专题的第四小节了。本节内容主要讲Math类和System类&#xff0c; 内容包括Math类介绍、Math类常用方法、System类介绍&#xff0c;System类常用方法。该小节内容基本不涉…

【教程】你现在还不知道微软的New Bing?你out了,快点进来看

哈喽啊&#xff0c;大家好&#xff0c;好久不见&#xff0c;我是木易巷&#xff01; 不禁感叹&#xff0c;AI人工智能时代真的已经来临&#xff01; 目前&#xff0c;谷歌和微软就各自面向大众的产品发布了重大公告。谷歌推出了一款名为Bard实验性对话式 AI 服务&#xff0c;而…

Python开发入门之了解Python高阶函数

上段时间有小伙伴询问&#xff1a;高阶函数的问题&#xff0c;今天小编就带大家一起来看一看&#xff1a; 一、什么是高阶函数? 高阶函数是在Python中一个非常有用的功能函数&#xff0c;所谓高阶函数就是一个函数可以用来接收另一个函数作为参数&#xff0c;这样的函数叫做…

扬帆优配|数字经济刮起“东风”,龙头晋级7连板

今日两市共40只涨停股&#xff0c;主要集中于数字经济、6G板块&#xff0c;上一个交易日涨停股为29股&#xff1b;除掉18只ST股及3只一字板新股&#xff0c;共19股涨停。另外&#xff0c;4股封板未遂&#xff0c;整体封板率为83%。 6股封单金额超亿元 从收盘涨停板封单量来看&…

第54章 图片URL的后端获取

1 注意&#xff1a; 在.NetCore WebApi框架中&#xff0c;在默认情况下由于没有集成“UseStaticFiles”内置管道中间件方法&#xff0c;如果想要通过图片URL显示图片&#xff0c;由会显示“404”错误&#xff0c;必须先把“UseStaticFiles”内置管道中间件方法集成到.NetCore W…

安装MySQL数据库8.0服务实例

前言 之前尝试去安装了MySQL5.7的社区版本&#xff0c;今天来安装MySQL8.0的版本&#xff0c;并且以两种方式进行安装&#xff0c;一个是通过RPM包的安装&#xff0c;另一个则是编译的方式。 一. 前期准备 查看服务器IP [rootlocalhost ~]# hostname -I 192.168.161.166 19…

js中window自带的四舍五入toFixed方法中的坑以及解决办法

Hello&#xff0c;各位&#xff0c;我胡汉三~啊呸&#xff0c;我又回来啦&#xff0c;还改了名&#xff0c;换了头像&#xff0c;哈哈哈&#xff01;时隔这么长时间不更新了&#xff0c;太忙了&#xff0c;平时笔记都记在了自己的电脑上&#xff0c;从今天起&#xff0c;继续更…

vxe-grid 全局自定义filter过滤器,支持字典过滤

一、vxe-table的全局筛选器filters的实现 官网例子&#xff1a;https://vxetable.cn/#/table/renderer/filter 进入之后&#xff1a;我们可以参照例子自行实现&#xff0c;也可以下载它的源码&#xff0c;进行调整 下载好后并解压&#xff0c;用vscode将解压后的文件打开。全局…

CRM系统中的营销自动化能解决什么问题

CRM客户管理系统营销自动化的范围远远超出了人们的认知。许多人认为它只是自动化完成重复和乏味的任务来减少营销人员的工作量。虽然这确实占了很大一部分&#xff0c;但它真正的价值在于提高潜客转化&#xff0c;增加业务收入。那么&#xff0c;什么是CRM系统营销自动化&#…

195、【动态规划】AcWing —— 91. 最短Hamilton路径(C++版本)

题目描述 原题链接&#xff1a;91. 最短Hamilton路径 解题思路 动态规划五步曲&#xff1a; &#xff08;1&#xff09;dp[i][j]含义&#xff1a; 到达点j并且状态为i时&#xff0c;具有的最短路径长度&#xff0c;其中状态j用状态压缩二进制的方式表示。j中从0-n-1位分别对…

【玩转c++】List讲解和模拟底层实现

本期主题&#xff1a;list的讲解和模拟实现博客主页&#xff1a;小峰同学分享小编的在Linux中学习到的知识和遇到的问题小编的能力有限&#xff0c;出现错误希望大家不吝赐1.list的介绍和使用1.1.list的介绍1.list是可以在常数范围内在任意位置进行插入和删除的序列式容器&…