【Go函数详解】三、匿名函数和闭包

2024-08-28 01:28
文章标签 go 函数 详解 匿名 闭包

本文主要是介绍【Go函数详解】三、匿名函数和闭包,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一、匿名函数的定义与使用
  • 二、匿名函数与闭包
    • 1. 闭包概念
    • 2. 闭包特点
    • 3. 闭包的实现原理
    • 4. 闭包的注意事项
      • 4.1 内存泄漏
      • 4.2 竞态条件
  • 三、匿名函数的常见使用场景
    • 1. 保证局部变量的安全性
    • 2. 将匿名函数作为函数参数
    • 3. 将匿名函数作为函数返回值


一、匿名函数的定义与使用

匿名函数时一种没有指定函数名的函数声明方式(与之相对的,有名字的函数被称为具名函数),在很多编程语言中都有实现和支持。

func(a, b int) int { return a + b
}

Go匿名函数也可以赋值给一个变量或者直接执行:

// 1、将匿名函数赋值给变量
add := func(a, b int) int {return a + b
}// 调用匿名函数 add
fmt.Println(add(1, 2))  // 2、定义时直接调用匿名函数
func(a, b int) {fmt.Println(a + b)
} (1, 2) 

为什么可以将匿名函数赋值给一个普通变量呢?以下解析

二、匿名函数与闭包

回答上面的问题需要了解闭包

1. 闭包概念

闭包(Closure)是指一个函数包含了它外部作用域中的变量,即使在外部作用域结束后,这些变量依然可以被内部函数访问和修改。闭包使得函数可以“记住”外部作用域的状态,这种状态在函数调用之间是保持的。
闭包的核心概念是函数内部可以引用外部作用域的变量,即使在函数内部外部作用域已经结束。
简单来说,【闭】的意思是【封闭外部状态】,即使外部状态已经失效,闭包内部依然保留了一份从外部引用的变量。

2. 闭包特点

  1. 函数可以在定义的作用域之外被调用,仍然可以访问外部作用域的变量。
  2. 外部作用域中的变量不会被销毁,直到闭包不再引用它们。
  3. 多个闭包可以共享同一个外部作用域的变量。

3. 闭包的实现原理

Go语言中的闭包是通过函数值(Function Value) 实现的。在Go语言中,函数不仅是代码,还是数据,可以像其他类型的值一样被传递、赋值和操作。当一个函数内部引用了外部作用域的变量时,Go编译器会生成一个闭包实例,将外部变量的引用与函数代码绑定在一起。

基本闭包

func makeCounter() func() int {count := 0return func() int {	//若这里是非匿名函数,那编译报错count++return count}
}counter := makeCounter()
fmt.Println(counter()) // 输出 1
fmt.Println(counter()) // 输出 2

makeCounter 函数返回一个匿名函数,这个匿名函数持有了外部变量 count 的引用。每次调用 counter() 时,都会访问和修改外部作用域的 count 变量。

闭包是一种函数对象,可以持有外部变量的状态。支持闭包的语言将函数视为第一类对象,使得函数具有与其他数据类型(整型、字符串、数组、切片、字典、结构体等)相同的地位,可以赋值给变量,也可以作为参数传递给其他函数,还能在运行时被函数动态地创建和返回。

4. 闭包的注意事项

4.1 内存泄漏

由于闭包持有外部作用域的变量引用,如果闭包一直被引用,外部作用域的变量不会被销毁,可能会导致内存泄漏。在使用闭包时,需要注意外部作用域变量的生命周期

4.2 竞态条件

在并发编程中,由于多个goroutine可以共享闭包中的变量,可能会引发竞态条件和数据不一致问题。在并发场景下使用闭包时,需要保证变量的访问是安全的。

三、匿名函数的常见使用场景

1. 保证局部变量的安全性

匿名函数内部声明的局部变量无法从外部修改,从而确保了安全性(类似类的私有属性):

var j int = 1f := func() {var i int = 1fmt.Printf("i, j: %d, %d\n", i, j)
}f()  //i, j: 1, 1
j += 2
f()  //i, j: 1, 3

在上面的示例中,匿名函数引用了外部变量,所以同时也是个闭包,变量 f 指向的闭包引用了局部变量 i 和 j,i 在闭包内部定义,其值被隔离,不能从外部修改,而变量 j 在闭包外部定义,所以可以从外部修改,闭包持有的是它的引用。

2. 将匿名函数作为函数参数

匿名函数除了可以赋值给普通变量外,还可以作为参数传递到函数中进行调用,就像普通数据类型一样:

add := func(a, b int) int {return a + b
}// 将函数类型作为参数
func(call func(int, int) int) {fmt.Println(call(1, 2))
}(add)

当我们将函数声明数据类型时,需要严格指定每个参数和返回值的类型,这才是一个完整的函数类型,因此 add 函数对应的函数类型是 func(int, int) int。

也可以将第二个匿名函数提取到 main 函数外,成为一个具名函数 handleAdd,然后定义不同的加法算法实现函数,并将其作为参数传入 handleAdd:

func main() {...// 普通的加法操作add1 := func(a, b int) int {return a + b}// 定义多种加法算法base := 10add2 := func(a, b int) int {return a * base + b}handleAdd(1, 2, add1)  // 3handleAdd(1, 2, add2)  // 12
}// 将匿名函数作为参数
func handleAdd(a, b int, call func(int, int) int) {fmt.Println(call(a, b))
}

就可以通过一个函数执行多种不同加法实现算法,提升了代码的复用性。

3. 将匿名函数作为函数返回值

// 将函数作为返回值类型
func deferAdd(a, b int) func() int {return func() int {return a + b}
}func main() {...// 此时返回的是匿名函数addFunc := deferAdd(1, 2)// 这里才会真正执行加法操作fmt.Println(addFunc())
}

在上面这个示例代码中,调用 deferAdd 函数返回的是一个匿名函数,但是这个匿名函数引用了外部函数传入的参数,因此形成闭包,只要这个闭包存在,这些持有的参数变量就一直存在,即使脱离了 deferAdd 函数的作用域,依然可以访问它们。
另外调用 deferAdd 方法时并没有执行闭包,只有运行 addFunc() 时才会真正执行闭包中的业务逻辑(这里是加法运算),因此,我们可以通过将函数返回值声明为函数类型来实现业务逻辑的延迟执行,让执行时机完全掌握在开发者手中。

这篇关于【Go函数详解】三、匿名函数和闭包的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1113335

相关文章

MySQL数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

MyBatis常用XML语法详解

《MyBatis常用XML语法详解》文章介绍了MyBatis常用XML语法,包括结果映射、查询语句、插入语句、更新语句、删除语句、动态SQL标签以及ehcache.xml文件的使用,感兴趣的朋友跟随小... 目录1、定义结果映射2、查询语句3、插入语句4、更新语句5、删除语句6、动态 SQL 标签7、ehc

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1

一文详解Python如何开发游戏

《一文详解Python如何开发游戏》Python是一种非常流行的编程语言,也可以用来开发游戏模组,:本文主要介绍Python如何开发游戏的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、python简介二、Python 开发 2D 游戏的优劣势优势缺点三、Python 开发 3D

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

Redis 基本数据类型和使用详解

《Redis基本数据类型和使用详解》String是Redis最基本的数据类型,一个键对应一个值,它的功能十分强大,可以存储字符串、整数、浮点数等多种数据格式,本文给大家介绍Redis基本数据类型和... 目录一、Redis 入门介绍二、Redis 的五大基本数据类型2.1 String 类型2.2 Hash