【泛型进阶】Go泛型应用之工厂方法+泛型的使用演化

  大家好,我是Coder哥,今天我们来聊一下泛型,go 泛型十年磨一剑,十年说长也不长,毕竟柯南还是小学生,说短也不短,如今鸣人都俩孩子了,哈哈哈。

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

  1. 接口是怎么实现泛化编程的。
  2. 泛型是怎么解决接口的局限性的。
  3. 泛型使用的最佳时机。
  4. 关于功能设计的简单建议。

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

接口实现泛化编程

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

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

比如如下一个需求:

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

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

关于功能设计的简单建议

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

最后

  感谢各位能看到最后,希望本篇的内容对你有帮助,有什么意见或者建议可以留言一起讨论,看到后第一时间回复,也希望大家能给个赞,你的赞就是我写文章的动力,再次感谢。

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

本文正在参加技术专题18期-聊聊Go语言框架

猜你喜欢

转载自juejin.im/post/7120104935953285134
今日推荐