Golang 泛型的简单使用
前情提要
美国时间2022年3月15日,Go核心团队官宣了Go 1.18版本正式版的发布,其官博称之为“整个Go社区的一个巨大的里程碑”。在这个版本中,Go核心团队做了Go语言开源以来的最大一次语法特性变更——增加了对泛型(generics)
的支持。
动手尝试
升级go 1.18
前往官网下载最新版本
Downloads - The Go Programming Language (google.cn)
需要卸载旧版本再安装新版本
修改VSCODE配置
VSCODE 可能会提示泛型语法只有 go 为 1.18 才可以使用
悬停红色波浪线,选择快速修复,执行提供的命令。重启VSCODE。
由于VSCODE go staticheck 会对go的泛型语法报黄色警告,我们需要修改 VSCODE 拓展设置,搜索 Lint Tool,将默认修改为 golint。
学习泛型
我们可以先参考 Go by Example 给出的例子
Go by Example 中文版: 范型 (gobyexample-cn.github.io)
package main
import "fmt"
func MapKeys[K comparable, V any](m map[K]V) []K {
r := make([]K, 0, len(m))
for k := range m {
r = append(r, k)
}
return r
}
K,V 两种类型,要求K特征是能比较,V特征是全部,相当于 interface{}
type List[T any] struct {
head, tail *element[T]
}
type element[T any] struct {
next *element[T]
val T
}
List
是一个 具有任意类型值的单链表。
func (lst *List[T]) Push(v T) {
if lst.tail == nil {
lst.head = &element[T]{val: v}
lst.tail = lst.head
} else {
lst.tail.next = &element[T]{val: v}
lst.tail = lst.tail.next
}
}
调用范型函数的时候, 我们经常可以使用类型推断。 注意,当调用 MapKeys
的时候, 我们不需要为 K
和 V
指定类型,编译器会自动进行类型推断
fmt.Println("keys m:", MapKeys(m))
尝试泛型
我们经常需要程序对数据集合执行操作,例如选择满足给定条件的全部item,或通过自定义函数将全部item映射到一个新的集合。
在其它语言中,通常会使用泛型数据结构和算法。
- Index、Include 方法参数为 comparable 类型,因为comparable特征的类型才可以进行比较
- Any、All、Filter、Map 方法参数为 any 类型,使用了组合函数,可以传入一个方法。
package main
import (
"fmt"
"strings"
)
func Index[T comparable](vs []T, item T) int {
for i, v := range vs {
if v == item {
return i
}
}
return -1
}
func Include[T comparable](vs []T, item T) bool {
return Index(vs, item) > 0
}
func Any[T any](vs []T, f func(T) bool) bool {
for _, v := range vs {
if f(v) {
return true
}
}
return false
}
func All[T any](vs []T, f func(T) bool) bool {
for _, v := range vs {
if !f(v) {
return false
}
}
return true
}
func Filter[T any](vs []T, f func(T) bool) []T {
res := make([]T, 0)
for _, v := range vs {
if f(v) {
res = append(res, v)
}
}
return res
}
func Map[T any, K any](vs []T, f func(T) K) []K {
res := make([]K, len(vs))
for i, v := range vs {
res[i] = f(v)
}
return res
}
func main() {
fmt.Println("Golang 泛型的初步学习")
ints := []int{10, 20, 30, 40, 50}
intCompare := func(t int) bool {
if t > 30 {
return true
}
return false
}
fmt.Println(ints)
fmt.Println("查找20下标", Index(ints, 20))
fmt.Println("是否存在100", Include(ints, 100))
fmt.Println("至少有一个大于30", Any(ints, intCompare))
fmt.Println("全部大于30", All(ints, intCompare))
fmt.Println("筛选大于30", Filter(ints, intCompare))
fmt.Println("放大一倍", Map(ints, func(t int) int { return t * 2 }))
textCompare := func(t string) bool {
if len(t) > 5 {
return true
}
return false
}
fruits := []string{"Apple", "Banana", "Orange"}
fmt.Println("查找Orange下标", Index(fruits, "Orange"))
fmt.Println("是否存在Apple", Include(fruits, "Apple"))
fmt.Println("至少有一个元素长度大于5", Any(fruits, textCompare))
fmt.Println("全部元素长度大于5", All(fruits, textCompare))
fmt.Println("筛选长度大于5", Filter(fruits, textCompare))
fmt.Println("每个元素重复一遍", Map(fruits, func(t string) string { return strings.Repeat(t, 2) }))
newS := Map(fruits, func(t string) string { return strings.Repeat(t, 2) })
fmt.Printf("%p %p", fruits, newS)
}
Golang 泛型的简单使用的更多相关文章
- golang实现的简单优先队列
下面是golang实现的简单优先队列,参考信息可以查看https://golang.org/pkg/container/heap/或者https://golang.google.cn/pkg/cont ...
- TODO:Golang UDP连接简单测试慎用Deadline
TODO:Golang UDP连接简单测试慎用Deadline UDP 是User Datagram Protocol的简称, 中文名是用户数据报协议,是OSI(Open System Interco ...
- java 泛型的简单使用
effecttive java一直推荐使用泛型,简单的看了一下泛型的使用 package cn.com.fzk; import java.util.ArrayList; import java.uti ...
- 数据结构和算法(Golang实现)(1)简单入门Golang-前言
数据结构和算法在计算机科学里,有非常重要的地位.此系列文章尝试使用 Golang 编程语言来实现各种数据结构和算法,并且适当进行算法分析. 我们会先简单学习一下Golang,然后进入计算机程序世界的第 ...
- 数据结构和算法(Golang实现)(2)简单入门Golang-包、变量和函数
包.变量和函数 一.举个例子 现在我们来建立一个完整的程序main.go: // Golang程序入口的包名必须为 main package main // import "golang&q ...
- 数据结构和算法(Golang实现)(3)简单入门Golang-流程控制语句
流程控制语句 计算机编程语言中,流程控制语句很重要,可以让机器知道什么时候做什么事,做几次.主要有条件和循环语句. Golang只有一种循环:for,只有一种判断:if,还有一种特殊的switch条件 ...
- 数据结构和算法(Golang实现)(4)简单入门Golang-结构体和方法
结构体和方法 一.值,指针和引用 我们现在有一段程序: package main import "fmt" func main() { // a,b 是一个值 a := 5 b : ...
- 数据结构和算法(Golang实现)(5)简单入门Golang-接口
接口 在Golang世界中,有一种叫interface的东西,很是神奇. 一.数据类型 interface{} 如果你事前并不知道变量是哪种数据类型,不知道它是整数还是字符串,但是你还是想要使用它. ...
- 数据结构和算法(Golang实现)(6)简单入门Golang-并发、协程和信道
并发.协程和信道 Golang语言提供了go关键字,以及名为chan的数据类型,以及一些标准库的并发锁等,我们将会简单介绍一下并发的一些概念,然后学习这些Golang特征知识. 一.并发介绍 我们写程 ...
随机推荐
- 串和KMP算法
一.串 串是由零个或多个字符串组成的有限序列 (一).串的定义 定长顺序存储 特点:每个串变量分配一个固定长度的存储区,即定长数组 定义: #define MAXLEN 255 typedef str ...
- python2,3的区别
python2和python3的区别: 1. python 2x:源代码不规范,源码重复多 python 3x:源码 清晰,简单,优美 2.python 2x:默认编码是ASCII码 编码更改: ...
- C#: .net序列化及反序列化 [XmlElement(“节点名称”)] [XmlAttribute(“节点属性”)] (下篇)
介绍 XML 序列化 .NET Framework 开发员指南 序列化是将对象转换为容易传输的格式的过程.例如,可以序列化一个对象,然后使用 HTTP 通过 Internet 在客户端和服务器之间 ...
- CF736D Permutations(伴随矩阵)
CF736D Permutations(伴随矩阵) Luogu 题解时间 首先把边直接放进邻接矩阵, 很明显行列式的奇偶和方案数的奇偶一样. 设 $ A_{ i , j } $ 为矩阵的该行列的余子式 ...
- 安卓开发学习-app结构学习
开发软件:Android Studio 全局分析 gradle与idea是AS自动生成的文件 buid是编译时生成的文件 gradle里面包含gradle wrapper配置文件 gitignore是 ...
- 什么时候用断言(assert)?
答:断言在软件开发中是一种常用的调试方式,很多开发语言中都支持这种机制.一般来说,断言用于保证程序最基本.关键的正确性.断言检查通常在开发和测试时开启.为了保证程序的执行效率,在软件发布后断言检查通常 ...
- redis 过期键的删除策略?
1.定时删除:在设置键的过期时间的同时,创建一个定时器 timer). 让定时器在键 的过期时间来临时,立即执行对键的删除操作. 2.惰性删除:放任键过期不管,但是每次从键空间中获取键时,都检查取得的 ...
- Python - 数据存储与数据库简介
- MCU选型
含义: MCU(Micro Controller Unit)中文名称为微控制单元,又称单片微型计算机(Single Chip Microcomputer),是指随着大规模集成电路的出现及其发展,将计算 ...
- carsim2016事件如何设置
#carsim2016事件设置# 完成以下功能:车速低于60km/h时,加速,设置节气门开度为0.8,制动主斜体样式缸压力设为0:车速高于120km/h时,制动,设置节气门开度为0,制动主缸压力设置为 ...