文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Go泛型应用工厂方法及泛型使用

2024-04-02 19:55

关注

前言

由于网上关于泛型使用的文章太多了,这里就不聊怎么使用泛型了,今天我们结合工厂方法+泛型方法来看一下泛型到底是如何在业务场景中使用的。本文涉及到的点如下:

话不多说,我们开始吧,具体泛型怎么使用及语法,请自行查阅相关资料。

接口实现泛化编程

平时我们编写结构体和方法的时候,一般是使用具体的类型:要么是基本类型,要么是自定义类型。但是如果要编写可以应用于多种类型的代码的时候,那么这种限制对程序的束缚就会比较大。

那么我们想编出一些泛化的方法和接口,怎么办呢? 这个时候我们想到了接口,如果方法的参数是一个接口,而不是一个结构体,这样对程序的限制就会放开了许多。因为任何实现该接口的结构体都可以作为该方法的接口参数,这样就可以保证后面在添加其他同类功能的时候,只需实现这个接口就可以满足需求了。

比如如下一个需求:

定义两个手机品牌结构体华为,苹果 ,并打印出各自品牌的名字。保证程序的扩展性,后面我们可能还要加入小米

import "fmt"
//手机统一接口
type Phone interface {
  PrintBrand()
}
​
type HuaweiPhone struct {
}
func (hw *HuaweiPhone) PrintBrand() {
  fmt.Printf("品牌名字:华为")
}
​
type Iphone struct {
}
func (ip *Iphone) PrintBrand() {
  fmt.Printf("品牌名字:苹果")
}
//统一打印方法
func PrintBrand(phone Phone) {
  phone.PrintBrand()
}
func main() {
  hw := HuaweiPhone{}
  ip := Iphone{}
  PrintBrand(ip)
  PrintBrand(hw)
}

如上面代码,我们定义了两个手机品牌的结构体,我们想打印各个手机的品牌名字,需要调用统一打印方法就可以了,如果后面添加其他品牌的话,我们只需要实现Phone这个接口就可以,如下添加小米手机品牌:

type XiaomiPhone struct {
}
func (xm XiaomiPhone) PrintBrand() {
  fmt.Printf("品牌名字:小米")
}

以上代码我们可以看到,通过接口也可以定义一些泛化的行为。

工厂+泛型来实现更通用的泛化编程

可是有时候,即便我们使用了接口,对程序的约束依然还是很强,因为一旦我们指明了具体的接口,就会要求我们必须使用特定的接口。而我们希望编写更通用的代码,要使代码能够应用于某种不具体的类型,而不是具体的一个接口或者结构体。这个要怎么办呢?

比如,我们基于以上的需求继续加需求

由于我们对接的品牌增大到20种,除了上面三种还有 魅族、三星、诺基亚、中兴。。。。等等

这个时候我们基于当前的代码已经不能满足,那么我们想到了工厂设计模式,在工厂中用泛型来泛化所有的类型,我们通过传入类型名字来打印出具体的品牌名。我们引入工厂模式继续优化我们的代码,

如下:

var cache sync.Map
//工厂方法 可以传入任意的类型
func PhoneFactory[T any]() (t *T) {
  target := reflect.TypeOf(t)
  v, ok := cache.Load(t)
  if ok {
    return v.(*T)
  }
  v = new(T)
  v, _ = cache.LoadOrStore(target, v)
  return v.(*T)
}
func main() {
  PrintBrand(PhoneFactory[Iphone]())
  PrintBrand(PhoneFactory[HuaweiPhone]())
  PrintBrand(PhoneFactory[XiaomiPhone]())
}

代码中我们编写了个工厂方法,泛型类型为 any, 接收任意的类型,在工厂中我们创建对象返回相应的类型并缓存类型对象防止重复创建。这样我们后面再加其他类别的时候可以通过这个工厂方法来统一的创建,我们还可以通过反射在创建前后根据业务需要做一些操作。

泛型使用的最佳时机

泛型的加入,无疑增加了代码的复杂度,那么我们使用泛型的最佳时机是什么时候呢?

Go 泛型主要设计者 Ian Lance Taylor 给出了简要的泛型使用方针,当开发者发现自己多次编写完全相同的代码,而这些副本之间的唯一区别仅在于使用了不同类型,这时候便可以考虑使用类型参数。换句话说,即开发者应避免使用类型参数,直到发现自己要多次编写完全相同的代码。

关于功能设计的简单建议

比如说上面的业务,其实我们开始设计的时候设计到接口层面就可以了,如果一开始就引入工厂方法,其实这算是过度设计,我们设计一个功能的原则是,抓住上下文,适度设计,因为一旦我们投入了过多的精力到灵活设计上,势必会影响本应该完成的需求。同时,过多的功能会引入更多潜在的问题,而修复问题也会耗费我们的时间和精力。而且在当前这个敏捷开发的时代,更是如此。

最后

为了提高可阅读性,以上代码都是以最简单的方式呈现的,实际业务远比这要复杂的多,这里只是提供一种方向。

到此这篇关于Go泛型应用工厂方法及泛型使用的文章就介绍到这了,更多相关Go泛型应用内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯