Go语言开发小技巧&易错点100例(十一)

简介: Go语言开发小技巧&易错点100例(十一)


本期看点(技巧类用【技】表示,易错点用【易】表示)

  • Go函数式编程【技】
  • 不建议map使用指针类型作为Key【易】
  • 直接使用值为nil的slice和map【易】

正文开始:

Go函数式编程

函数式编程是一种编程范式。函数式编程语言最重要的基础是λ演算,λ演算的函数可以接受函数当作输入(参数)和输出(返回值)。与指令式编程相比,函数式编程强调函数的计算比指令的执行重要。与过程化编程相比,函数式编程里函数的计算可随时调用。

此外,在函数式编程中,函数是一等公民,这意味着它们可以绑定到名称(包括本地标识符),作为参数传递,并从其他函数返回,就像任何其他数据类型一样。这允许以声明性和可组合的风格编写程序,其中小功能以模块化方式组合。

我们来展示一下Go语言的函数式编程(大家可以猜想一下这段代码的运行结果):

func PlayFunc(str string, fn func() error) error {
  fmt.Println(str)
  defer func() {
    fmt.Println("defer 1 ...")
  }()
  defer func() {
    fmt.Println("defer 2 ...")
  }()
  return fn()
}
func main() {
  err := PlayFunc("string ...", func() error {
    fmt.Println("func ...")
    return nil
  })
  fmt.Println(err)
}

答案:

string ...
func ...
defer 2 ...
defer 1 ...
<nil>
不建议map使用指针类型作为Key

在Go语言中,指针类型不能作为map的键(key)的主要原因是因为指针的值是动态的,并且可能会发生变化。当使用指针作为map的键时,如果两个指针指向同一个内存地址,它们被认为是相等的,但是如果指针所指向的值发生变化,那么这两个指针就不再相等了。

举个例子:

type Student struct {
   Id   string
   Name string
}
func TestMapPointKey(t *testing.T) {
   m := make(map[*Student]struct{})
   m[&Student{Id: "1", Name: "zs"}] = struct{}{}
   _, ok := m[&Student{Id: "1", Name: "zs"}]
   fmt.Println(ok) // false
}

为了解决这个问题,Go语言规定map的键必须是不可变(immutable)的类型,例如基本类型(如整数、字符串等),或者具有只读属性的复合类型(如数组、结构体等)。这些类型的值在创建后就不能被修改,因此它们可以作为map的键使用。

比如这样:

func TestMap(t *testing.T) {
   m := make(map[Student]struct{})
   m[Student{Id: "1", Name: "zs"}] = struct{}{}
   _, ok := m[*&Student{Id: "1", Name: "zs"}]
   fmt.Println(ok) // true
}

基本数据类型下的指针类型也会存在这个问题:

func TestMapInt(t *testing.T) {
   m := make(map[*int]struct{})
   p := 1
   m[&p] = struct{}{}
   p1 := 1
   _, ok := m[&p1]
   fmt.Println(ok) // false 
   m2 := make(map[int]struct{})
   p2 := 1
   m2[p2] = struct{}{}
   p3 := 1
   _, ok = m2[p3]
   fmt.Println(ok) // true
}

总结起来,Go语言中指针类型不能作为map的键是因为指针的值是动态的,可能会发生变化,而map的键需要是不可变的类型。

直接使用值为nil的slice和map
func TestEmptyMap(t *testing.T) {
  var m map[string]struct{}
  m["name"] = struct{}{}
}

这段代码是一个Go语言的测试函数,但是它有一个错误。声明了一个名为m的map,该map的键是字符串类型,而值是空结构体类型(struct{})。由于m是一个空的map(即它还没有任何键值对),因此不能直接赋值。这将导致运行时错误。为了修复这个错误,需要首先为map m分配一个值(比如 m = make(map[string]struct{})),然后再尝试插入键值对。

相关文章
|
7天前
|
存储 JSON 监控
Viper,一个Go语言配置管理神器!
Viper 是一个功能强大的 Go 语言配置管理库,支持从多种来源读取配置,包括文件、环境变量、远程配置中心等。本文详细介绍了 Viper 的核心特性和使用方法,包括从本地 YAML 文件和 Consul 远程配置中心读取配置的示例。Viper 的多来源配置、动态配置和轻松集成特性使其成为管理复杂应用配置的理想选择。
24 2
|
5天前
|
Go 索引
go语言中的循环语句
【11月更文挑战第4天】
13 2
|
5天前
|
Go C++
go语言中的条件语句
【11月更文挑战第4天】
16 2
|
8天前
|
监控 Go API
Go语言在微服务架构中的应用实践
在微服务架构的浪潮中,Go语言以其简洁、高效和并发处理能力脱颖而出,成为构建微服务的理想选择。本文将探讨Go语言在微服务架构中的应用实践,包括Go语言的特性如何适应微服务架构的需求,以及在实际开发中如何利用Go语言的特性来提高服务的性能和可维护性。我们将通过一个具体的案例分析,展示Go语言在微服务开发中的优势,并讨论在实际应用中可能遇到的挑战和解决方案。
|
5天前
|
Go
go语言中的 跳转语句
【11月更文挑战第4天】
13 4
|
5天前
|
JSON 安全 Go
Go语言中使用JWT鉴权、Token刷新完整示例,拿去直接用!
本文介绍了如何在 Go 语言中使用 Gin 框架实现 JWT 用户认证和安全保护。JWT(JSON Web Token)是一种轻量、高效的认证与授权解决方案,特别适合微服务架构。文章详细讲解了 JWT 的基本概念、结构以及如何在 Gin 中生成、解析和刷新 JWT。通过示例代码,展示了如何在实际项目中应用 JWT,确保用户身份验证和数据安全。完整代码可在 GitHub 仓库中查看。
18 1
|
7天前
|
Go 调度 开发者
探索Go语言中的并发模式:goroutine与channel
在本文中,我们将深入探讨Go语言中的核心并发特性——goroutine和channel。不同于传统的并发模型,Go语言的并发机制以其简洁性和高效性著称。本文将通过实际代码示例,展示如何利用goroutine实现轻量级的并发执行,以及如何通过channel安全地在goroutine之间传递数据。摘要部分将概述这些概念,并提示读者本文将提供哪些具体的技术洞见。
|
11天前
|
JavaScript Java Go
探索Go语言在微服务架构中的优势
在微服务架构的浪潮中,Go语言以其简洁、高效和并发处理能力脱颖而出。本文将深入探讨Go语言在构建微服务时的性能优势,包括其在内存管理、网络编程、并发模型以及工具链支持方面的特点。通过对比其他流行语言,我们将揭示Go语言如何成为微服务架构中的一股清流。
102 53
|
10天前
|
Ubuntu 编译器 Linux
go语言中SQLite3驱动安装
【11月更文挑战第2天】
32 7
|
10天前
|
关系型数据库 Go 网络安全
go语言中PostgreSQL驱动安装
【11月更文挑战第2天】
41 5