《Go语言圣经》函数值、匿名函数递归与可变参数

函数值(Function Values)

在 Go 语言中,函数被视为第一类值(first-class values),这意味着它们可以像其他值一样被操作:拥有类型、赋值给变量、作为参数传递给其他函数或作为返回值。函数值的调用方式与普通函数相同。

func square(n int) int { return n * n }
func negative(n int) int { return -n }
func product(m, n int) int { return m * n }f := square
fmt.Println(f(3)) // 输出: 9f = negative
fmt.Println(f(3))     // 输出: -3
fmt.Printf("%T\n", f) // 输出: func(int) intf = product // 编译错误: 无法将 func(int, int) int 赋值给 func(int) int

函数类型的零值是 nil,调用值为 nil 的函数会导致 panic 错误:

var f func(int) int
f(3) // 此处 f 为 nil,会触发 panic

函数值可以与 nil 比较,但函数值之间不可比较,也不能作为 map 的键:

var f func(int) int
if f != nil {f(3)
}

匿名函数递归(Recursive Anonymous Functions)

当需要定义递归调用的匿名函数时,必须先声明变量并指定类型,再将匿名函数赋值给该变量。这是因为 Go 编译器需要在函数体内部解析函数类型。

以下示例展示了如何使用递归匿名函数进行拓扑排序(Topological Sort):

// prereqs 记录了每个课程的前置课程
var prereqs = map[string][]string{"algorithms": {"data structures"},"calculus": {"linear algebra"},"compilers": {"data structures","formal languages","computer organization",},"data structures":       {"discrete math"},"databases":             {"data structures"},"discrete math":         {"intro to programming"},"formal languages":      {"discrete math"},"networks":              {"operating systems"},"operating systems":     {"data structures", "computer organization"},"programming languages": {"data structures", "computer organization"},
}func main() {for i, course := range topoSort(prereqs) {fmt.Printf("%d:\t%s\n", i+1, course)}
}func topoSort(m map[string][]string) []string {var order []stringseen := make(map[string]bool)// 声明递归函数类型var visitAll func(items []string)// 赋值匿名函数visitAll = func(items []string) {for _, item := range items {if !seen[item] {seen[item] = truevisitAll(m[item]) // 递归调用order = append(order, item)}}}var keys []stringfor key := range m {keys = append(keys, key)}sort.Strings(keys)visitAll(keys)return order
}

关键点

  1. 必须先声明 visitAll 变量并指定类型 func(items []string)
  2. 再将匿名函数赋值给 visitAll
  3. 函数体内部可正确解析 visitAll 的类型

可变参数(Variadic Functions)

可变参数函数可以接收任意数量的指定类型参数,在参数列表最后一个类型前加 ... 表示:

func sum(vals ...int) int {total := 0for _, val := range vals {total += val}return total
}fmt.Println(sum())           // 输出: 0
fmt.Println(sum(3))          // 输出: 3
fmt.Println(sum(1, 2, 3, 4)) // 输出: 10

在函数体内部,可变参数被视为切片类型(如 []int)。若要传递现有切片给可变参数函数,需在切片后加 ...

values := []int{1, 2, 3, 4}
fmt.Println(sum(values...)) // 输出: 10

可变参数函数与以切片为参数的函数类型不同:

func f(...int) {}
func g([]int) {}fmt.Printf("%T\n", f) // 输出: func(...int)
fmt.Printf("%T\n", g) // 输出: func([]int)

可变参数函数常用于格式化字符串,例如:

func errorf(linenum int, format string, args ...interface{}) {fmt.Fprintf(os.Stderr, "Line %d: ", linenum)fmt.Fprintf(os.Stderr, format, args...)fmt.Fprintln(os.Stderr)
}linenum, name := 12, "count"
errorf(linenum, "undefined: %s", name) // 输出: Line 12: undefined: count

其中 interface{} 表示最后一个参数可接收任意类型。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/diannao/87704.shtml
繁体地址,请注明出处:http://hk.pswp.cn/diannao/87704.shtml
英文地址,请注明出处:http://en.pswp.cn/diannao/87704.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vtk和opencv和opengl直接的区别是什么?

简介 VTK、OpenCV 和 OpenGL 是三个在计算机图形学、图像处理和可视化领域广泛使用的工具库,但它们在功能、应用场景和底层技术上存在显著差异。以下是它们的核心区别和特点对比: 1. 核心功能与定位 工具核心功能主要应用领域VTK (Visualization Toolk…

最新豆包大模型发布!火山引擎推出Agent开发新范式

Datawhale大会 2025火山引擎 Force 原动力大会 6月11日-12日,北京国家会议中心人山人海,2025 火山引擎 Force 原动力大会如约而至。 作为开发者社区的一员,这场大会上的一系列新发布让我们感受到了:这个 Agent 技术落地元年的关键…

RFC4291-IPv6地址架构解说

RFC 4291 是由互联网工程任务组(IETF)发布的关于 IPv6 地址架构 的标准文档。 该文档详细定义了 IPv6 地址的格式、类型、表示方法以及分配方式。 以下是对 RFC 4291 中 IPv6 地址架构的全面解析,包括地址格式、类型、表示方法、特殊地址以…

简单对比 **HTTP**、**MQTT** 和 **CoAP** 这三种通信协议

对比 HTTP、MQTT 和 CoAP 这三种通信协议,从 消息结构、资源占用、安全性 等方面进行全面分析。 🌐 HTTP vs MQTT vs CoAP 对比 特性HTTPMQTTCoAP协议层级应用层基于 TCP应用层基于 TCP / WebSocket应用层基于 UDP (也支持 TCP)消息模式请求/响应 (客户…

【Dify 案例】【自然语言转SQL案例】【五】【实战二】【财务管理查询商品信息数据】

援引实战一,进行数据业务处理化 1.开始 2.自然语言转SQL的工具 3.参数提取器 4.SQL查询

FPGA基础 -- Verilog语言要素之标识符

一、什么是标识符(Identifier) 在 Verilog 中,标识符是用户定义的名字,用于标识模块、变量、端口、函数、任务、参数、宏定义等各种语言要素。 就像 C 语言的变量名、函数名一样,Verilog 中的标识符为 HDL 代码提供了…

Tomcat双击startup.bat闪退的解决方法

首先需要确认java环境是否配置正确,jdk是否安装正确 winR打开cmd,输入该命令 java -version 出现对应的版本就说明jdk配置正确 如果没有,则参考jdk的安装及配置 如果以上都没有问题,就继续排查 确认Tomcat的环境变量配置 概…

计算机基础(三):深入解析Java中的原码、反码、补码

计算机基础系列文章 计算机基础(一):ASCll、GB2312、GBK、Unicode、UTF-32、UTF-16、UTF-8深度解析 计算机基础(二):轻松理解二进制、八进制、十进制和十六进制 计算机基础(三):深入解析Java中的原码、反码、补码 目录 引言一、 基础概念&…

phpstudy无法启动mysql,一启动就关闭,完美解决

phpstudy无法启动mysql,一启动就关闭,完美解决 phpstudy的mysql无法启动,一启动就关闭如何解决。 问题出现的原因:phpstudy自带的mysql,可能与之前单独安装的mysql发生冲突。(之前安装的mysql已经占用3306端口) 解决方…

mysql中的<>和!=

在MySQL中&#xff0c;<> 运算符表示 不等于。它与 ! 运算符功能完全相同&#xff0c;都是用于比较两个表达式是否不相等。 SELECT * FROM table_name WHERE column_name <> value;当 column_name 的值不等于 value 时&#xff0c;返回该行当值相等或为 NULL 时&a…

C#学习日记

命名空间 知识点一 命名空间基本概念 概念 命名空间是用来组织和重用代码的 作用 就像是一个工具包&#xff0c;类就像是一件一件的工具&#xff0c;都是申明在命名空间中的 知识点二 命名空间的使用 基本语法 namespace 命名空间名 {类类 } namespace MyGame {class GameO…

第八十二篇 大数据开发基础:树形数据结构深度解析与实战指南(附创新生活案例)

目录 一、树的本质&#xff1a;层次化数据组织二、生活中的树形智慧&#xff1a;无处不在的层次案例1&#xff1a;图书馆图书分类系统案例2&#xff1a;电商平台商品类目树案例3&#xff1a;城市行政区域划分 三、大数据中的核心树结构1. B树&#xff1a;数据库索引的脊梁2. 决…

从0开始学计算机视觉--Day1--计算机视觉的起源

我们经常能听到计算机视觉这个词语&#xff0c;像数字图像处理&#xff0c;算法设计&#xff0c;深度学习等领域。但很少有人会先去了解清楚这门知识&#xff0c;而是用到什么再学什么&#xff0c;虽然这在项目进度上能节省不少时间&#xff0c;但有时候囫囵吞枣式地学习容易落…

简单的 ​Flask​ 后端应用

from flask import Flask, request, jsonify, session import os app Flask(__name__) app.secret_key os.urandom(24) users { 123: admin, admin: admin } # 登录接口 app.route(/login, methods[POST]) def login(): data request.get_json() username data.get(usern…

spring-webmvc @PathVariable 典型用法

典型用法 基础用法 GetMapping("/users/{id}") public String getUser(PathVariable Long id) {return "User ID: " id; } 请求&#xff1a;/users/1001 输出&#xff1a;User ID: 1001---- GetMapping("/users/{userId}/orders/{orderId}") …

LVS+Keepliaved高可用群集

目录 keepalived双击热备基础知识1.keepallived概述及安装keepalived的热备方式 2.使用keepalived实现双机热备 案例1.基础主备调度器环境配置2.配置主调度器3.配置从调度器4.配置两台节点服务器5.测试 keepalived双击热备基础知识 Keepalived 起初是专门针对 LVS 设计的一款强…

在Unreal Engine 5(UE5)中,Get PlayerPawn和Get PlayerController的区别以及如何计算玩家和目标之间的距离。

一、两者区别 在Unreal Engine 5&#xff08;UE5&#xff09;中&#xff0c;获取玩家的位置信息通常有两种方式&#xff1a;通过PlayerPawn或通过PlayerController。具体使用哪一个取决于你想要获取的是哪个实体的位置。 1.Get Player Pawn&#xff1a; PlayerPawn是玩家实际…

linux线程同步

互斥锁 同步与互斥概述** 现代操作系统基本都是多任务操作系统&#xff0c;即同时有大量可调度实体在运行。在多任务操作系统中&#xff0c;同时运行的多个任务可能&#xff1a; 都需要访问/使用同一种资源 多个任务之间有依赖关系&#xff0c;某个任务的运行依赖于另一个任…

Spring 的IoC 和 AOP

第一部分&#xff1a;关于 IoC (控制反转) 1. 核心思想 (What & Why) 首先&#xff0c;我会先解释 IoC 的核心思想&#xff0c;而不是直接讲技术。 “IoC&#xff0c;即控制反转&#xff0c;它是一种重要的设计思想&#xff0c;而不是一个具体的技术。它的核心是将传统上…

[实战] Windows 文件读写函数 `ReadFile()` 和 `WriteFile()` 的阻塞与非阻塞操作详解(含完整C语言示例)

Windows 文件读写函数 ReadFile() 和 WriteFile() 的阻塞与非阻塞操作详解&#xff08;含完整C语言示例&#xff09; 在 Windows 平台进行文件或设备&#xff08;如串口、管道&#xff09;编程时&#xff0c;ReadFile() 和 WriteFile() 是最常用的两个 API 函数。它们既可以以…