当前位置: 首页 > news >正文

Go语言---闭包

文章目录

  • 基本介绍
  • 基本概念
  • 闭包示例
  • 闭包的核心特性
  • 闭包的典型使用场景
    • 1. 计数器/生成器模式
    • 2. 函数工厂
    • 3.中间件模式
  • 闭包捕获的外部变量存储位置
    • 存储机制详解
  • 被闭包捕获的外部变量的修改影响范围
    • 1. 多个闭包共享同一个外部变量(会影响)
    • 2. 每次调用生成独立的闭包实例(不会影响)
    • 3.关键区分点
  • 闭包底层原理
  • 注意事项
    • 1. 循环中的闭包陷阱
    • 2. 并发安全问题
    • 3. 性能

基本介绍

闭包(Closure)是Go语言中一个重要的特性,它允许函数访问并操作其外部作用域中的变量。闭包在Go中广泛用于实现函数式编程模式、状态保持和回调等场景。

基本概念

闭包是一个函数值,它引用了函数体之外的变量。这个函数可以访问并修改这些外部变量,也就是说函数"绑定"了这些变量。这个函数和这些变量共同组成闭包。

闭包示例

func main() {x := 10// 这是一个闭包,它捕获了外部变量xadd := func(y int) int {return x + y}fmt.Println(add(5)) // 输出15x = 20fmt.Println(add(5)) // 输出25,闭包能看到x的变化
}

在这里插入图片描述

闭包的核心特性

1、变量捕获:闭包可以捕获并持有外部作用域的变量。

2、状态保持:被捕获的变量在闭包调用间保持其状态。

3、独立实例:每次创建闭包都会生成一个新的独立环境。

闭包的典型使用场景

1. 计数器/生成器模式

counter代码解析:定义了一个名为counter的函数,没有参数,返回值为fun() int。

func counter() func() int {i := 0return func() int {i++return i}
}func main() {c1 := counter()fmt.Println(c1()) // 1fmt.Println(c1()) // 2c2 := counter()fmt.Println(c2()) // 1 (新的实例)
}

2. 函数工厂

两个实例,两个闭包的string不相互影响。

func makeGreeter(prefix string) func(string) string {return func(name string) string {return prefix + ", " + name}
}func main() {hello := makeGreeter("Hello")hi := makeGreeter("Hi")fmt.Println(hello("Alice")) // Hello, Alicefmt.Println(hi("Bob"))      // Hi, Bob
}

3.中间件模式

func loggerMiddleware(next http.HandlerFunc) http.HandlerFunc {return func(w http.ResponseWriter, r *http.Request) {start := time.Now()next(w, r)log.Printf("%s %s took %v", r.Method, r.URL.Path, time.Since(start))}
}

闭包捕获的外部变量存储位置

在Go语言中,被闭包函数捕获的外部变量存储在堆(heap)上,而不是栈(stack)上。这是Go闭包实现的一个重要特性。

存储机制详解

Go编译器会进行逃逸分析,确定变量的存储位置。当变量被闭包引用时,编译器会判定它"逃逸"到了堆上。这是为了保证变量的生命周期能够延长到闭包的使用期。
编译器会将被捕获的变量和闭包函数打包成一个结构体,这个结构体会分配在堆内存中闭包函数通过这个结构体来访问被捕获的变量。

被闭包捕获的外部变量的修改影响范围

被闭包捕获的外部变量的修改是否会影响所有实例,取决于闭包的创建方式。具体分为两种情况:
1、多个闭包共享同一个外部变量(会影响)。
2、每次调用生成独立的闭包实例(不会影响)。

1. 多个闭包共享同一个外部变量(会影响)

当多个闭包捕获的是同一个外部变量时,修改该变量会影响所有相关的闭包实例。

func main() {var i int = 0// 两个闭包捕获同一个i变量incr := func() { i++ }get := func() int { return i }fmt.Println(get()) // 0incr()fmt.Println(get()) // 1 (两个闭包看到的是同一个i)
}

2. 每次调用生成独立的闭包实例(不会影响)

当每次函数调用都创建新的变量和闭包时,各个闭包实例拥有自己的变量副本,互不影响。

func counter() func() int {i := 0 // 每次调用counter()都会创建新的ireturn func() int {i++return i}
}func main() {c1 := counter() // 有自己的ic2 := counter() // 有另一个独立的ifmt.Println(c1()) // 1 (c1的i)fmt.Println(c1()) // 2fmt.Println(c2()) // 1 (c2的i,不受c1影响)fmt.Println(c1()) // 3 (c1的i继续独立递增)
}

3.关键区分点

情况变量声明位置影响范围示例
共享变量闭包外部声明所有闭包实例共享多个闭包捕获同一个包级/函数级变量
独立变量闭包创建函数内部每个闭包实例独立像counter()工厂函数那样每次创建新变量

闭包底层原理

Go的闭包实现基于以下几点:

1、闭包函数会持有对外部变量的引用。

2、编译器会将闭包和它引用的外部变量打包成一个结构体。

3、当闭包被调用时,它会通过这个结构体访问外部变量。

底层实现示例(概念模型):

// 编译器生成的类似结构(实际实现更复杂)
type closureStruct struct {i int  // 被捕获的变量// 可能还有其他捕获的变量
}func counter() func() int {c := &closureStruct{i: 0}  // 分配在堆上return func() int {c.i++return c.i}
}

注意事项

1. 循环中的闭包陷阱

func main() {var funcs []func()for i := 0; i < 3; i++ {// 错误写法:所有闭包共享同一个ifuncs = append(funcs, func() { fmt.Println(i) })}for _, f := range funcs {f() // 全部输出3,不是预期的0,1,2}// 正确写法1:通过参数传递for i := 0; i < 3; i++ {i := i // 创建局部变量副本funcs = append(funcs, func() { fmt.Println(i) })}// 正确写法2:立即执行for i := 0; i < 3; i++ {func(i int) {funcs = append(funcs, func() { fmt.Println(i) })}(i)}
}

2. 并发安全问题

当多个goroutine访问同一个闭包变量时,需要加锁:

func safeCounter() func() int {var i intvar mu sync.Mutexreturn func() int {mu.Lock()defer mu.Unlock()i++return i}
}

3. 性能

闭包会延长被捕获变量的生命周期,可能导致内存占用增加,在性能敏感的场景需要谨慎使用。

http://www.xdnf.cn/news/1033705.html

相关文章:

  • golang字符串拼接
  • 【MFC 突然被问到,怎么实现一个星星按钮】原来问的是继承xs
  • CTF题目:Apache Flink目录遍历漏洞实战及CVE-2020-17519漏洞分析
  • 标准库转hal库
  • Kafka - 并发消费拉取数据过少故障分析
  • PyTorch张量操作中dim参数的核心原理与应用技巧:
  • 【机械视觉】Halcon—【十三、实例找各个区域面积和中心点】
  • 大模型成长过程-预训练tokenizer
  • 2.5 Rviz使用教程
  • 人工智能学习13-Numpy-规律数组生成
  • pytorch基本运算-梯度运算:requires_grad_(True)和backward()
  • 多个项目的信息流如何统一与整合
  • Spring AI Chat Tool Calling 指南
  • MySQL使用EXPLAIN命令查看SQL的执行计划
  • 13.20 LangChain多链协同架构实战:LanguageMentor实现67%对话连贯性提升
  • [每周一更]-(第144期):Go 定时任务的使用:从基础到进阶
  • mysql 创建大写字母的表名失败
  • HarmonyOS 组件复用 指南
  • React中使用Day.js指南
  • ABC410 : F - Balanced Rectangles
  • MIB 树的来源与实现深度解析
  • 计算机网络学习笔记:运输层概述UDP、TCP对比
  • Arduino入门教程​​​​​​​:4、打印字符到电脑
  • 疫菌QBD案例
  • Gartner《Build Scalable Data Products With This Step-by-Step Framework》学习报告
  • Linux系统安装MongoDB 8.0流程
  • 树莓派智能小车红外避障实验指导书
  • 当遇到“提交失败:404”的问题时,通常表明前端请求的URL无法正确匹配到后端的Servlet或资源。
  • 区间合并:区间合并问题
  • 前端与协议