Go 泛型解密:从基础到实战的全方位解析

Go泛型解密

file

一、概述

泛型编程是计算机科学中一个相当重要的概念,广泛应用于各种编程语言和框架中。在Go语言中,泛型的讨论和实现也走了一段相对漫长的路。这一路上既有激烈的讨论,也有种种的尝试和迭代。本节将对泛型的基础概念进行深入分析,并探究其在Go中的历史与现状。

什么是泛型

泛型,又称为"参数多态",是一种允许你编写出可以处理不同数据类型(而非单一数据类型)的代码的程序设计范式。泛型有助于提高代码复用性,增加类型安全性,以及有时还能优化性能。

例如,在其他支持泛型的语言如Java、C#中,我们可以很方便地定义一个可以处理任何数据类型的列表:

List<T> list = new ArrayList<T>();

在Go语言中,借助于泛型,我们也可以实现类似的功能:

type List[T any] struct {// ...
}

这里的T就是一个类型参数,any是一个类型约束,表示T可以是任何类型。

泛型在Go中的历史与进展

泛型在Go语言的历史中一直是一个备受关注的话题。Go语言最初的设计哲学是追求简单和高效,因此在最初版本中并没有加入泛型。然而随着社群和企业对更灵活、更强大功能的追求,泛型逐渐显露出其不可或缺的重要性。

  • Go1.x时代:在Go 1.x的版本中,泛型并没有被纳入。开发者通常使用interface{}和类型断言来模拟泛型,但这种方式有其局限性,如类型安全性不足、性能开销等。

  • Go 2的设计草案:在Go 2的设计阶段,泛型成为了社区最关注的一项特性。经过多次设计与反馈循环,最终泛型被列为Go 2的核心改进之一。

  • 实验和反馈:在多次的实验和社区反馈后,Go团队逐渐明确了泛型的设计目标和具体语法。例如,类型参数、类型约束等成为了泛型实现的关键元素。

  • 正式发布:经过多年的讨论和改进,泛型最终在Go的某个版本(例如Go 1.18)中正式发布。

file


二、为什么需要泛型

file


泛型编程作为一种编程范式,不仅仅存在于Go语言中。从C++的模板到Java的泛型,从Python的类型提示到Rust的泛型,这一概念在软件工程和编程语言设计中有着广泛的应用和深远的影响。那么,为什么我们需要泛型呢?本节将从三个主要方面进行详细解释:类型安全、代码复用和性能优化。

类型安全

弱类型的弊端

在没有泛型的情况下,Go语言中的interface{}经常被用作通用类型,这样可以接受任何类型的参数。然而,这样做会失去类型检查的好处。

func Add(a, b interface{}) interface{} {return a.(int) + b.(int)  // 需要类型断言,且不安全
}

上面的代码示例中,ab的类型在运行时才会被检查,这就增加了出错的可能性。

强类型的优势

泛型通过在编译期进行类型检查,来解决这个问题。

func Add[T Addable](a, b T) T {return a + b  // 类型安全
}

这里,Addable是一个类型约束,只允许那些满足某些条件的类型(比如,可以进行加法操作的类型)作为泛型参数。

代码复用

无泛型的局限性

在没有泛型的情况下,如果我们想为不同类型实现相同的逻辑,通常需要写多个几乎相同的函数。

func AddInts(a, b int) int {return a + b
}func AddFloats(a, b float64) float64 {return a + b
}

泛型的通用性

有了泛型,我们可以写出更加通用的函数,而无需牺牲类型安全性。

func Add[T Addable](a, b T) T {return a + b
}

性能优化

一般而言,泛型代码由于其高度抽象,可能会让人担心性能损失。但事实上,在Go语言中,泛型的实现方式是在编译期间生成特定类型的代码,因此,性能损失通常是可控的。

编译期优化

由于Go编译器在编译期会为每个泛型参数生成具体的实现,因此,运行时不需要进行额外的类型检查或转换,这有助于优化性能。

// 编译期生成以下代码
func Add_int(a, b int) int {return a + b
}func Add_float64(a, b float64) float64 {return a + b
}

三、Go泛型的基础

Go语言在版本1.18之后正式引入了泛型,这是一个让许多Go开发者期待已久的功能。本节将深入讲解Go泛型的基础,包括类型参数、类型约束,以及泛型在函数和数据结构中的应用。

类型参数

基础语法

在Go中,泛型的类型参数通常使用方括号进行声明,紧随函数或结构体名称之后。

func Add[T any](a, b T) T {return a + b
}

这里,T 是一个类型参数,并且使用了 any 约束,意味着它可以是任何类型。

多类型参数

Go泛型不仅支持单一的类型参数,你还可以定义多个类型参数。

func Pair[T, U any](a T, b U) (T, U) {return a, b
}

在这个例子中,Pair 函数接受两个不同类型的参数 a 和 b,并返回这两个参数。

类型约束

内建约束

Go内置了几种类型约束,如 any,表示任何类型都可以作为参数。

func PrintSlice[T any](s []T) {for _, v := range s {fmt.Println(v)}
}

自定义约束

除了内置约束,Go还允许你定义自己的约束。这通常是通过接口来实现的。

type Addable interface {int | float64
}func Add[T Addable](a, b T) T {return a + b
}

这里,Addable 是一个自定义的类型约束,只允许 int 或 float64 类型。

泛型函数与泛型结构体

泛型函数

我们已经看到了几个泛型函数的例子,它们允许你在多种类型上执行相同的逻辑。

func Max[T comparable](a, b T) T {if a > b {return a}return b
}

泛型结构体

除了函数,Go也支持泛型结构体。

type Box[T any] struct {Content T
}

这里,Box 是一个泛型结构体,它有一个 Content 字段,类型为 T

泛型方法

在泛型结构体中,你还可以定义泛型方法。

func (b Box[T]) Empty() bool {return b.Content == nil
}

四、Go泛型高级特性

在前一节中,我们探讨了Go泛型的基础,包括类型参数、类型约束以及泛型函数和泛型结构体。本节将聚焦于Go泛型的高级特性,涵盖类型列表、泛型与接口的交互,以及在现实世界中的应用场景。

类型列表

类型组合

Go泛型允许使用类型组合,在一个约束中指定多种允许的类型。

type Numeric interface {int | float64
}func Sum[T Numeric](s []T) T {var total Tfor _, v := range s {total += v}return total
}

在这个例子中,Numeric 约束允许 int 和 float64 类型,使得 Sum 函数能在这两种类型的切片上进行操作。

多约束

Go也支持多约束的概念,即一个类型需要满足多个接口。

type Serializable interface {json.Marshaler | xml.Marshaler
}

泛型与接口的交互

泛型作为接口的方法

你可以在接口中定义包含泛型的方法。

type Container[T any] interface {Add(element T)Get(index int) T
}

使用接口约束泛型

与泛型约束相似,接口也可以用于约束泛型类型。

func PrintIfHuman[T HumanLike](entity T) {if entity.IsHuman() {fmt.Println(entity)}
}

这里,HumanLike 是一个接口,IsHuman 是它的一个方法。

泛型在实际应用中的场景

泛型数据结构

在实际应用中,泛型通常用于实现通用的数据结构,比如链表、队列和堆栈。

type Stack[T any] struct {elements []T
}func (s *Stack[T]) Push(element T) {s.elements = append(s.elements, element)
}func (s *Stack[T]) Pop() T {element := s.elements[len(s.elements)-1]s.elements = s.elements[:len(s.elements)-1]return element
}

用于算法实现

泛型也在算法实现中有广泛应用,特别是那些不依赖于具体类型的算法。

func Sort[T Ordered](arr []T) []T {// 排序算法实现
}

五、Go泛型实战举例

在前几节中,我们已经深入探讨了Go泛型的基础和高级特性。现在,我们将通过一系列具体的实战示例来演示如何在实际项目中使用Go泛型。

泛型实现一个简单的数组列表

定义

一个泛型数组列表需要能够进行添加、删除和读取元素。我们可以使用泛型来定义这样一个数据结构。

type ArrayList[T any] struct {items []T
}

实例

下面,我们实现了添加元素和读取元素的方法。

func (al *ArrayList[T]) Add(item T) {al.items = append(al.items, item)
}func (al *ArrayList[T]) Get(index int) (T, error) {if index < 0 || index >= len(al.items) {return zero(T), errors.New("Index out of bounds")}return al.items[index], nil
}
输入和输出

假设我们有一个 ArrayList[int],我们添加数字 1 和 2,然后尝试获取索引为 1 的元素。

al := &ArrayList[int]{}
al.Add(1)
al.Add(2)
element, err := al.Get(1) // 输出:element=2, err=nil

使用泛型构建缓存系统

定义

缓存系统通常需要存储任意类型的数据并能够在给定的时间内检索它们。我们可以使用泛型和Go的内建 map 类型来实现这一点。

type Cache[T any] struct {store map[string]T
}

实例

我们实现了一个简单的 Set 和 Get 方法来操作缓存。

func (c *Cache[T]) Set(key string, value T) {c.store[key] = value
}func (c *Cache[T]) Get(key string) (T, bool) {value, exists := c.store[key]return value, exists
}
输入和输出

考虑一个场景,我们需要缓存字符串。

c := &Cache[string]{store: make(map[string]string)}
c.Set("name", "John")
value, exists := c.Get("name") // 输出:value="John", exists=true

泛型实现快速排序

定义

快速排序是一种高效的排序算法。由于它不依赖于具体的数据类型,因此很适合使用泛型来实现。

实例

以下是一个使用泛型的快速排序算法实现。

func QuickSort[T comparable](arr []T) {if len(arr) < 2 {return}pivot := arr[len(arr)/2]var less, greater []Tfor _, x := range arr {if x < pivot {less = append(less, x)} else if x > pivot {greater = append(greater, x)}}QuickSort(less)QuickSort(greater)// 合并结果// ...
}
输入和输出

如果我们有一个整数切片 [3, 1, 4, 1, 5, 9, 2, 6, 5],使用 QuickSort 后,我们应得到 [1, 1, 2, 3, 4, 5, 5, 6, 9]

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

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

相关文章

基于SSM+Vue的咖啡销售系统

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;Vue 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#xff1a;是 目录…

菜鸟智谷产业园正式开园!入驻企业可享受“城区+园区”双重政策扶持

位于杭州未来科技城的菜鸟智谷产业园正式开园了! 杭州未来科技城发布新电商产业加速“百千万亿”计划,菜鸟智谷产业园发布“产业赋能计划”,并设立天猫品牌孵化基地,落户中国(杭州)跨境电商综试区一带一路电商运营中心,引入悦汇跨境产业创新基金……10月17日,杭州未来科技城(海…

SpringBoot+SpringSecurity项目的创建

首先创建一个springboot项目&#xff0c;注意springboot要求jdk版本在17以上 等项目构建完成之后&#xff0c;在pom文件中加入SpringSecurity的依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-securi…

Linux基础命令(二)

1.ls命令 命令格式&#xff1a; ls -a -l -h Linux路径 &#xff08;1&#xff09; 直接使用ls&#xff0c;不加选项和参数。 ls 将会以平铺形式&#xff0c;列出当前文件夹下的内容&#xff1a; &#xff08;2&#xff09;加上参数 ls / 将会列出/目录下的内容。 -a选项…

JVM垃圾回收算法介绍

堆的分代和区域 &#xff08;年轻代&#xff09;Young Generation&#xff08;eden、s0、s1 space&#xff09; Minor GC &#xff08;老年代&#xff09;Old Generation &#xff08;Tenured space&#xff09; Major GC|| Full GC &#xff08;永久代&#xff09;Permanent…

关注用户信息卡片

效果展示 CSS 知识点 box-shadow 属性回顾CSS 变量回顾 实现页面整体布局 <div class"card"><div class"box"><!-- 视频 --><div class"vide_box"><video src"user.mp4" type"video/mp4" aut…

UDP与TCP协议

很抱歉&#xff0c;我之前写好的UDP与TCP文章不小心被删了&#xff0c;所以&#xff0c;这篇文章只有一半&#xff0c;后面我会尽快补全。 在完成HTTPS的学习后&#xff0c;我们就完成了应用层的所有讲解&#xff0c;下面我们开始讲解传输层&#xff0c;这一层常用的协议为TCP…

什么是CSGO大行动,2023年CSGO大行动时间预测

什么是CSGO大行动&#xff0c;2023年CSGO大行动时间预测 什么是CSGO大行动&#xff0c;2023年CSGO大行动时间预测 那天群里在提大行动&#xff0c;不明所以的新同学在问&#xff0c;什么是大行动&#xff0c;是不是官方红锁大行动要来了&#xff1f;当然不是&#xff0c;别自己…

面试十分钟不到就被赶出来了,问的实在是太变态了...

从外包出来&#xff0c;没想到算法死在另一家厂子 自从加入这家公司&#xff0c;每天都在加班&#xff0c;钱倒是给的不少&#xff0c;所以也就忍了。没想到8月一纸通知&#xff0c;所有人不许加班&#xff0c;薪资直降30%&#xff0c;顿时有吃不起饭的赶脚。 好在有个兄弟内…

云计算:掌控未来,一触即发!

&#x1f389;&#x1f389;欢迎来到我的CSDN主页&#xff01;&#x1f389;&#x1f389; &#x1f3c5;我是尘缘&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f449;点击这里&#xff0c;就可以查看我的主页啦&#xff01;&#x1f447;&#x…

springboot时间管理系统springboot47

大家好✌&#xff01;我是CZ淡陌。一名专注以理论为基础实战为主的技术博主&#xff0c;将再这里为大家分享优质的实战项目&#xff0c;本人在Java毕业设计领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目&#xff0c;希望你能有所收获&#xff0c;少走一些弯路…

从培训班出来之后找工作的经历,教会了我五件事.....

我是非计算机专业&#xff0c;由于专业不好实习急着就业有过一些失败的工作经历后&#xff0c;跑去参加培训进入IT这行的。 之前在报名学习软件测试之前我也很纠结&#xff0c;不知道怎么选择机构。后面看到有同学在知乎上分享自己的学习经历&#xff0c;当时对我的帮助很大。…

美芯片禁令再次扩大,波及英伟达、AMD以及intel等科技公司 | 百能云芯

拜登政府17日宣布&#xff0c;计划停止英伟达&#xff08;Nvidia&#xff09;、超微半导体以及英特尔等科技公司设计的先进AI芯片输出中国大陆&#xff0c;英伟达&#xff08;Nvidia&#xff09;昨日股价重挫4.68%至每股439.38美元&#xff1b;天风国际证券分析师郭明錤表示&am…

Elasticsearch介绍及插件head和kibana下载

目录标题 一、Elasticsearch介绍二、Elasticsearch下载三、Elasticsearch-head四、Elasticsearch-kibana 一、Elasticsearch介绍 Elasticsearch是什么? Elasticsearch 是一个基于Lucene的分布式搜索和分析引擎&#xff0c;ES是elaticsearch简写&#xff0c;Elasticsearch是一…

解决vue3 + vite + ts 中require失效的问题(require is not defind)

require is not defind因为require是属于webpack的方法&#xff0c;vite中找不到这个方法肯定报错 解决办法 通过vite官网了解到新的引入方式&#xff0c;我使用了其中一种 imgList: [{name: "lj",src: new URL(../../assets/img/applyList.png, import.meta.url).…

图像检索算法 计算机竞赛

文章目录 1 前言2 图像检索介绍(1) 无监督图像检索(2) 有监督图像检索 3 图像检索步骤4 应用实例5 最后 1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 图像检索算法 该项目较为新颖&#xff0c;适合作为竞赛课题方向&#xff0c;学长非常推荐&#xff…

.npmrc 使用详解

配置.npmrc之后需要&#xff1a; 清理项目目录中的 node _modules 目录(package-lock.json,umi)。清理 node cache: npm cache clear --force&#xff1b;{ 此步骤必须&#xff0c;主要是大家的电脑经过多年使用后&#xff0c;npm 配置比较混乱&#xff0c;为了避免或者减少配…

LLM 系列 | 21 : Code Llama实战(上篇) : 模型简介与评测

引言 小伙伴们好&#xff0c;我是《小窗幽记机器学习》的小编&#xff1a;卖热干面的小女孩。 个人CSDN首页&#xff1a;JasonLiu1919_面向对象的程序设计,深度学习,C-CSDN博客 今天开始以2篇小作文介绍代码大语言模型Code Llama。上篇主要介绍Code Llama的基本情况并基于Hug…

怎么多号发圈和批量加好友?

你知道怎么多号发圈和批量加好友吗&#xff1f; 我们都知道&#xff0c;微信号多&#xff0c;管理起来是一件相当麻烦的事。 那发圈和加好友&#xff0c;多号的话&#xff0c;那是相当大的工作量&#xff0c;那有没有什么办法可以多号同时进行发圈和加人吗&#xff1f; 当然有的…

Linux高性能服务器编程 学习笔记 第十六章 服务器调制、调试和测试

Linux平台的一个优秀特性是内核微调&#xff0c;即我们可以通过修改文件的方式来调整内核参数。 服务器开发过程中&#xff0c;可能会碰到意想不到的错误&#xff0c;一种调试方法是用tcpdump抓包&#xff0c;但这种方法主要用于分析程序的输入和输出&#xff0c;对于服务器的…