深度思考:为什么需要泛型?

简介: 深度思考:为什么需要泛型?

不知道大家平时在进行后端编程的时候有没有考虑过一个概念:泛型编程,就像面向对象、面向接口编程一样,很常用以致于用成为了大家广泛的习惯,在后端常用编程语言中,无论是Java、C++都支持泛型编程,而在2022年的3月份,随着Go1.18稳定版本的发布,Go自1.18版本起,也支持了“泛型(Generics)”这一特性,这同时也是Go语言在语法层面的一次重大改变。

虽然之前在使用Java进行编程时经常用到泛型,但是未曾思考过到底为什么需要泛型?没有泛型会怎样?泛型带来了什么作用?泛型的实现原理是怎样的?等等问题。

因为Go1.18版本发布已有几个月的时间,各个IDE也陆续支持Go语言泛型编码,因此也通过一些资料学习了Go语言泛型这个新特性,并且对此做了一些思考,想以一篇文章来向大家分享自己的思考经验和见解,同时也会以实际代码的方式使用Java、Go语言的泛型特性,剖析其原理,下面开始正文。

1 什么是泛型?

维基百科提到:最初泛型编程这个概念来自于缪斯·大卫和斯捷潘诺夫. 亚历山大合著的“泛型编程”一文。那篇文章对泛型编程的诠释是:“泛型编程的中心思想是对具体的、高效的算法进行抽象,以获得通用的算法,然后这些算法可以与不同的数据表示法结合起来,产生各种各样有用的软件”。说白了就是将算法与类型解耦,实现算法更广泛的复用。

在我看来,泛型是同接口类似,也是编程的一种规范也可以说是一种风格,泛型编程可以让开发者在编写代码时约束变量、容器、对象、结构体的类型,对类型清晰的掌握可以减少bug的产生,增强代码的可读性,让抽象变得更加具体和实用。基于泛型的程序,由于传入的参数不同,程序会实现不同的功能。这也被叫做一种多态现象,叫做参数化多态(Parametric Polymorphism)。

2 编程语言中泛型编程的实例

2.1 Java泛型编程

请移步这篇文章《玩转Java泛型》

2.2 Go泛型编程

package main
import "fmt"
type MyList[T any] struct {
   Items []Item[T]
}
type Item[T any] struct {
   Index int
   Value T
}
func (list *MyList[T]) AddItem(i T) {
   item := Item[T]{Value: i, Index: len(list.Items)}
   list.Items = append(list.Items, item)
}
func (list *MyList[T]) GetItem(index int) T {
   l := list.Items
   var val T
   for i := range l {
      if l[i].Index == index {
         val = l[i].Value
      }
   }
   return val
}
func (list *MyList[T]) Print() {
   for i := range list.Items {
      fmt.Println(list.Items[i])
   }
}
type MyHashMap[K comparable, V any] struct {
   Value map[K]V
}
func (m *MyHashMap[K, V]) SetValue(k K, v V) {
   m.Value[k] = v
}
func (m *MyHashMap[K, V]) GetValue(k K) V {
   return m.Value[k]
}
func (m *MyHashMap[K, V]) Print() {
   for k := range m.Value {
      fmt.Println(k, m.Value[k])
   }
}
func main() {
   list := MyList[int]{}
   list.AddItem(1)
   list.AddItem(2)
   item := list.GetItem(7)
   list.Print()
   hashMap := MyHashMap[string, int]{map[string]int{"A": 1, "B": 2}}
   hashMap.SetValue("s", 2)
   value := hashMap.GetValue("s")
   hashMap.Print()
}
复制代码

具体Go泛型编程内容可以看下这篇文章哈:《一文搞懂Go1.18泛型新特性》

3 为什么需要泛型?

回答这个问题之前,我们不妨思考下,在一些场景下如果没有泛型会怎样:

public class Main {
    static class Score {
        String name;
        int num;
        public Score(String name, int num) {
            this.name = name;
            this.num = num;
        }
    }
    public static int getSum(List<Score> scores) {
        int sum = 0;
        for (Score score : scores) {
            sum += score.num;
        }
        return sum;
    }
    public static void main(String[] args) {
        List<Score> scores = Arrays.asList(
                new Score("zs", 100),
                new Score("ls", 80),
                new Score("ww", 90.5) //编译不通过
        );
        int sum = getSum(scores);
        System.out.println(sum);
    }
}
复制代码

没有泛型时解决上述问题:

public class Main {
    static class Score {
        String name;
        int num;
        public Score(String name, int num) {
            this.name = name;
            this.num = num;
        }
    }
    static class Score2 {
        String name;
        float num;
        public Score2(String name, float num) {
            this.name = name;
            this.num = num;
        }
    }
    public static int getIntSum(List<Score> scores) {
        int sum = 0;
        for (Score score : scores) {
            sum += score.num;
        }
        return sum;
    }
    public static float getFloatSum(List<Score2> scores) {
        float sum = 0;
        for (Score2 score : scores) {
            sum += score.num;
        }
        return sum;
    }
    public static void main(String[] args) {
        List<Score> scores = Arrays.asList(
                new Score("zs", 100),
                new Score("ls", 80)
        );
        List<Score2> scores2 = Arrays.asList(
                new Score2("zs", 89.5f),
                new Score2("ls", 80.5f)
        );
        int sum = getIntSum(scores);
        float sum2 = getFloatSum(scores2);
        System.out.println(sum+sum2);
    }
}
复制代码

接下来我们引入泛型:

public class Main {
    static class Score<T> {
        String name;
        T num;
        public Score(String name, T num) {
            this.name = name;
            this.num = num;
        }
    }
    public static int getIntSum(List<Score<Integer>> scores) {
        int sum = 0;
        for (Score<Integer> score : scores) {
            sum += score.num;
        }
        return sum;
    }
    public static float getFloatSum(List<Score<Float>> scores) {
        float sum = 0;
        for (Score<Float> score : scores) {
            sum += score.num;
        }
        return sum;
    }
    public static void main(String[] args) {
        List<Score<Integer>> scores = Arrays.asList(
                new Score("zs", 100),
                new Score("ls", 80)
        );
        List<Score<Float>> scores2 = Arrays.asList(
                new Score("zs", 89.5f),
                new Score("ls", 80.5f)
        );
        int sum = getIntSum(scores);
        float sum2 = getFloatSum(scores2);
        System.out.println(sum+sum2);
    }
}
复制代码

所以,使用泛型的原因:

  • 泛化
  • 类型安全
  • 消除强制类型转换
  • 向后兼容

图示:

网络异常,图片无法展示
|


4 总结泛型的实现原理

大多数静态类型语言的泛型实现都是在编译期进行,也就是编译的前端实现,主要的技术包括类型擦除、具体化和基于元编程等进行的,比如Java的泛型就是基于类型擦除实现,在编译前端进行类型检查即可,编译之后的字节码不管有没有泛型都是一样的,运行时也是如此。而Go语言的泛型实现则不同,Go使用类似于具体化的方式实现泛型,就是在运行时使用类型信息,根据类型参数创建不同的具体类型的变量。

参考:

time.geekbang.org/column/arti…

baike.baidu.com/item/%E6%B3…

time.geekbang.org/column/arti…


相关文章
|
机器学习/深度学习 人工智能 城市大脑
阿里云 ET
阿里云 ET自制脑图, 由阿里云科学家团队研发的超级人工智能ET,是杭州城市大脑背后的人工智能中枢,也是阿里巴巴集团董事局主席马云眼中的下一代 CEO。阿里云 ET 拥有全球领先的人工智能技术。
1043 0
阿里云 ET
|
10月前
|
监控 数据可视化 项目管理
如何利用工作计划管理软件提升项目管理透明度?高效工具推荐与使用指南
在快速发展的商业环境中,项目管理工具已成为团队协作和工作计划管理的必备利器。本文推荐2024年5款最受欢迎的工作计划管理软件:板栗看板、Wrike、TeamGantt、Smartsheet和Podio。这些工具各具特色,能有效提升项目经理的工作效率,管理复杂的项目流程。无论是简洁高效的板栗看板,还是强调跨部门协作的Wrike,或是高度定制化的Podio,都能满足不同团队的需求。选择合适的工具,需综合考虑团队规模、项目复杂度等因素。
如何利用工作计划管理软件提升项目管理透明度?高效工具推荐与使用指南
|
存储 弹性计算 人工智能
阿里云Alex Chen:普惠计算服务,助力企业创新
本文整理自阿里云弹性计算产品线、存储产品线产品负责人陈起鲲(Alex Chen)在2024云栖大会「弹性计算专场-普惠计算服务,助力企业创新」中的分享。在演讲中,他分享了阿里云弹性计算,如何帮助千行百业的客户在多样化的业务环境和不同的计算能力需求下,实现了成本降低和效率提升的实际案例。同时,基于全面升级的CIPU2.0技术,弹性计算全线产品的性能、稳定性等关键指标得到了全面升级。此外,他还宣布了弹性计算包括:通用计算、加速计算和容器计算的全新产品家族,旨在加速AI与云计算的融合,推动客户的业务创新。
107134 10
|
C++
SDL基础使用02(加载bmp图片、纹理和渲染)
这篇文章介绍了如何使用SDL库在C++中加载和显示BMP图片,以及如何使用纹理和渲染器进行更高级的图形处理。
173 2
|
存储 算法 调度
FreeRTOS多任务系统
FreeRTOS多任务系统
438 0
|
搜索推荐 API
淘宝商品数据洞察:解锁精准营销新策略
在快速变化的商业环境中,高效的营销策略对企业至关重要。通过API获取淘宝APP的商品细节数据,企业可以精准分析产品特性、强化卖点,并制定灵活的价格策略。利用用户画像实现个性化营销,选择最佳渠道并优化内容,从而提升品牌影响力。这一方法不仅帮助企业抓住目标消费者,还能增强市场竞争力,促进长期发展。
|
SQL 数据库
SQL中CASE WHEN THEN ELSE END的用法详解
SQL中CASE WHEN THEN ELSE END的用法详解
2875 2
|
canal SQL 消息中间件
阿里Canal框架(数据同步中间件)初步实践
阿里Canal框架(数据同步中间件)初步实践
1172 92
阿里Canal框架(数据同步中间件)初步实践
|
Arthas Java 测试技术
Arthas中cat 、cls、echo、grep基础命令应用
通过本教程的操作,您可以体验如何Alibaba Cloud Linux  2.1903 LTS 64位操作系统的云服务器上学习cat 、cls、echo、grep基础命令教程用法。
|
存储 开发框架 监控
asp.net实验室信息管理LIMS系统源码
系统建立标准及项目库、产品分类库、检验项目库分别管理,并通过检验项目库的灵活应用和配置,用户在无需编程辅助的情况下,可灵活调整承检产品(样品)的分类、子类名称,可有效管理标准方法,可有效管理与标准对应的检验项目。产品(样品)库与标准方法(及检验项目库)可灵活的自由组合。检验项目库可在业务受理、项目检验等环节被方便的调用。
179 1