Go 并发实战核心编程【三】接口最佳实践

简介: Go 并发实战核心编程【三】接口最佳实践

接口最佳实践



sort包学习


1. sort对整形切片或者数组排序


平时写的整形切片排序:


var sl = []int{2, 4, 3, 1}
 sort.Ints(sl)
 fmt.Println(sl)


输出:[1 2 3 4]

我们看sort源码中有专门针对整形切片实现排序,例如:


sli := sort.IntSlice{2, 5, 3, 1}
 sli.Sort()
 fmt.Println(sli)


输出:[1 2 3 5]

我们看下源码实现:


// IntSlice attaches the methods of Interface to []int, sorting in increasing order.
type IntSlice []int
func (x IntSlice) Len() int           { return len(x) }
func (x IntSlice) Less(i, j int) bool { return x[i] < x[j] }
func (x IntSlice) Swap(i, j int)      { x[i], x[j] = x[j], x[i] }
// Sort is a convenience method: x.Sort() calls Sort(x).
func (x IntSlice) Sort() { Sort(x) }


我们发现IntSlice类型实现了三个方法,而且最后Sort方法调用了函数Sort,这个函数就是sort包中真正核心意义上的排序函数,底层是快速排序实现的,我们可以看下:


// Sort sorts data.
// It makes one call to data.Len to determine n and O(n*log(n)) calls to
// data.Less and data.Swap. The sort is not guaranteed to be stable.
func Sort(data Interface) {
 n := data.Len()
 quickSort(data, 0, n, maxDepth(n))
}


我们可以继续定位到quickSort,但是这里重点介绍接口最佳实践,所以不再深入快速排序内部逻辑,大家下去自行研究。


我们看到Sort函数参数是Interface接口,而这个接口定义如下:


type Interface interface {
 Len() int
 Less(i, j int) bool
 Swap(i, j int)
}

啊哈,看到这里我们发现IntSlice是实现了Interface接口,所以它可以调用上述Sort函数实现整形切片排序。


但是它这个排序是从小到大排序的,那如果从大到小实现排序呢?给各位留个思考题哈。


2. sort对结构体排序


因为对结构体的切片排序已经有相关函数实现了,比如看如下结构体:


// 学生
type StudentStruct struct {
 Name string
 Age int
}
type StudentSlice []StudentStruct
func (ss StudentSlice) Len() int {
 return len(ss)
}
// Less 按照年龄排序
func (ss StudentSlice) Less(i, j int) bool {
 return ss[i].Age < ss[j].Age
}
func (ss StudentSlice) Swap(i, j int) {
 ss[i], ss[j] = ss[j], ss[i]
}


用sort包中SliceStable函数实现结构体切片排序:


var students StudentSlice
 for i:=0; i<10; i++ {
  ss := StudentStruct{
   Name: fmt.Sprintf("姓名:%d", i),
   Age: rand.Intn(100),
  }
  students = append(students, ss)
 }
 sort.SliceStable(students, func(i, j int) bool {
  return students[i].Age < students[j].Age
 })
 for _, v:=range students {
  fmt.Println(v)
 }


输出:

{姓名:9 0}

{姓名:5 18}

{姓名:6 25}

{姓名:7 40}

{姓名:2 47}

{姓名:8 56}

{姓名:3 59}

{姓名:0 81}

{姓名:4 81}

{姓名:1 87}


我们看到按照年龄已经排序了,但是如果我们要实现自己的排序规则怎么办呢?

答案很简单,自己实现接口:


type StudentSlice []StudentStruct
func (ss StudentSlice) Len() int {
 return len(ss)
}
// Less 按照姓名从大到小排序
func (ss StudentSlice) Less(i, j int) bool {
 return ss[i].Name > ss[j].Name
}
func (ss StudentSlice) Swap(i, j int) {
 ss[i], ss[j] = ss[j], ss[i]
}
func main() {
 var students StudentSlice
 for i:=0; i<4; i++ {
  ss := StudentStruct{
   Name: fmt.Sprintf("姓名:%d", i),
   Age: rand.Intn(100),
  }
  students = append(students, ss)
 }
 fmt.Println("排序前")
 for _, v:=range students {
  fmt.Println(v)
 }
 fmt.Println("排序后")
 sort.Sort(students)
 for _, v:=range students {
  fmt.Println(v)
 }
}


输出:


排序前

{姓名:0 81}

{姓名:1 87}

{姓名:2 47}

{姓名:3 59}

排序后

{姓名:3 59}

{姓名:2 47}

{姓名:1 87}

{姓名:0 81}


我们看到排序已经是按照我们自己的排序规则排完序了,赞哦!


3. 小结


面向接口编程是一种很牛班的技术,它对底层设计者和开发者能力要求极高,所以一般复杂的底层infra建设都是有公司的大牛们担任,他们定义好接口之后对于技术小白来说你只要按照人家大牛的接口定义实现相应的方法即可,其他底层复杂的逻辑交给大牛们实现。当然如果有一天你写方法写累了,也想成为大牛搞搞底层建设,那么别着急,平时的日子里需要积累,相信吧,快乐的那一天将会来临!

相关文章
|
1月前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
2月前
|
存储 Rust Go
Go nil 空结构体 空接口有什么区别?
本文介绍了Go语言中的`nil`、空结构体和空接口的区别。`nil`是预定义的零值变量,适用于指针、管道等类型;空结构体大小为0,多个空结构体实例指向同一地址;空接口由`_type`和`data`字段组成,仅当两者均为`nil`时,空接口才为`nil`。
Go nil 空结构体 空接口有什么区别?
|
2月前
|
存储 负载均衡 监控
如何利用Go语言的高效性、并发支持、简洁性和跨平台性等优势,通过合理设计架构、实现负载均衡、构建容错机制、建立监控体系、优化数据存储及实施服务治理等步骤,打造稳定可靠的服务架构。
在数字化时代,构建高可靠性服务架构至关重要。本文探讨了如何利用Go语言的高效性、并发支持、简洁性和跨平台性等优势,通过合理设计架构、实现负载均衡、构建容错机制、建立监控体系、优化数据存储及实施服务治理等步骤,打造稳定可靠的服务架构。
55 1
|
2月前
|
Go 调度 开发者
探索Go语言中的并发模式:goroutine与channel
在本文中,我们将深入探讨Go语言中的核心并发特性——goroutine和channel。不同于传统的并发模型,Go语言的并发机制以其简洁性和高效性著称。本文将通过实际代码示例,展示如何利用goroutine实现轻量级的并发执行,以及如何通过channel安全地在goroutine之间传递数据。摘要部分将概述这些概念,并提示读者本文将提供哪些具体的技术洞见。
|
3月前
|
Java 大数据 Go
Go语言:高效并发的编程新星
【10月更文挑战第21】Go语言:高效并发的编程新星
65 7
|
2月前
|
并行计算 安全 Go
Go语言的并发特性
【10月更文挑战第26天】Go语言的并发特性
26 1
|
3月前
|
安全 程序员 Go
深入浅出Go语言的并发之道
在本文中,我们将探索Go语言如何优雅地处理并发编程。通过对比传统多线程模型,我们将揭示Go语言独特的goroutine和channel机制是如何简化并发编程,并提高程序的效率和稳定性。本文不涉及复杂的技术术语,而是用通俗易懂的语言,结合生动的比喻,让读者能够轻松理解Go语言并发编程的核心概念。
|
25天前
|
存储 监控 算法
员工上网行为监控中的Go语言算法:布隆过滤器的应用
在信息化高速发展的时代,企业上网行为监管至关重要。布隆过滤器作为一种高效、节省空间的概率性数据结构,适用于大规模URL查询与匹配,是实现精准上网行为管理的理想选择。本文探讨了布隆过滤器的原理及其优缺点,并展示了如何使用Go语言实现该算法,以提升企业网络管理效率和安全性。尽管存在误报等局限性,但合理配置下,布隆过滤器为企业提供了经济有效的解决方案。
72 8
员工上网行为监控中的Go语言算法:布隆过滤器的应用
|
1月前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
45 7
|
1月前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
112 71