Go语言程序设计-第5章--函数
Go语言程序设计-第5章–函数
5.1 函数声明
每个函数声明都包含一个名字、一个形参列表、一个可选的返回列表以及函数体:
func name(parameter-list) (result-list) {body
}
func add(x int, y int) int { return x + y}
func sub(x, y int) (z int) {z = x - y; return}
func first(x int, _ int) int { return x }
func zero(int, int) int {return 0}fmt.Printf("%T\n", add) // "func(int, int) int"
fmt.Printf("%T\n", sub) // "func(int, int) int"
fmt.Printf("%T\n", add) // "func(int, int) int"
fmt.Printf("%T\n", zero) // "func(int, int) int"
函数的类型称为函数签名。
实参是按值传递的。如果提供的实参包含引用类型,比如指针、slice、map、函数或者通道,那么当函数使用形参变量时就有可能间接地修改实参变量。
有些函数的声明没有函数体,说明这个函数使用除了 Go 以外的语言实现。
package math
func Sin(x float64) float64 // 使用汇编语言实现
5.2 递归
5.3 多返回值
函数可以有多个返回值。一个函数如果有命名的返回值,可以省略 return 语句的操作数,称为裸返回。
5.4 错误
Go 程序使用通常的控制流机制(比如if 和 return语句)应对错误。
5.4.1 错误处理策略
5.4.2 文件结束标识
EOF 定义:
package io
import "errors"var EOF = errors.New("EOF")
使用示例:
in := bufio.NewReader(os.Stdin)
for {r, _, err := in.ReadRune()if err == io.EOF {break // 结束读取}if err != nil {return fmt.Errorf("read failed: %v", err)}
}
5.5 函数变量
函数变量也有类型,可以赋给变量或者传递给其他函数,或者从其他函数中返回。
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
mt.Println(f(3)) // "9"
函数类型的零值是nil(空值),调用一个空的函数变量导致宕机。
5.6 匿名函数
strings.Map(func(r rune) rune) {return r + 1}, "HAL-9000")
函数里可以使用外层函数的变量。这些隐藏的变量引用就是我们把函数归类为引用类型而且函数变量无法进行比较的原因。函数变量类似于使用闭包方法实现的变量,Go 程序员通常把函数变量成为闭包。
func squares() func() int {var x intreturn func() int {x++return x * x}
}
func main() {f := squares()fmt.Println(f())fmt.Println(f())fmt.Println(f())fmt.Println(f())
}
输出:
1
4
9
16
警告:捕获迭代变量
var rmdirs []func()
for _, d := range tempDirs() {dir := d // 注意,这一行是必须的os.MkdirAll(dirkk, 0755)rmdirs = append(rmdirs, func() {os.RemoveAll(dir)})
}for _, rmdir := range rmdirs {rmdir() // 清理
}
为什么在循环体内将循环变量赋给一个新的局部变量 dir,而不是下面的版本。
var rmdirs []func()
for _, d := range tempDirs() {os.MkdirAll(dirkk, 0755)rmdirs = append(rmdirs, func() {os.RemoveAll(dir)})
}
原因是循环变量的作用域的规则限制。在上面的程序中,dir 在 for 循环引进的一个块作用域内进行声明。在循环里创建的所有函数变量共享相同的变量 – 一个可以访问的存储位置,而不是固定的值。dir 变量的值在不断地迭代中更新,因为当调用清理函数时,dir 变量是最后一次迭代时的值。我们用内部变量解决这个问题。
for _, dir := range tempDirs() {dir := dir // 声明内部 dir,并以外表 dir 初始化
}
5.7 变长函数(有可变的参数个数)
func sum(vals ...int) int {total := 0for _, val := range vals {total += val}return total
}
变长函数的类型和一个带有普通 slice 参数的函数类型不相同。
5.8 延迟函数调用
package ioutilfunc ReadFile(filename string)([]byte, error) {f, err := os.Open(filename)if err != nil {return nil, err}defer f.Closereturn ReadAll(f)
}
defer 语句,无论在正常的情况下,执行 return 语句或者函数执行完毕,还是在不正常的情况下,比如发生宕机,实际的调用推迟到包含 defer 语句的函数结束后才放行。
func bigSlowOperation() {defer traxce("bigSlowOperation")() // 别忘记这对圆括号time.Sleep(10 * time.Second)
}func trace(msg string) func() {start := time.Now()log.Printf("enter %s", msg)return func() { log.Printf("exit %s (%s)", msg, time.Since(start))}
}
延迟的匿名函数能够改变外层函数返给调用者的结果。
func triple(x int) (result int) {defer func() { result += x }()return x + x
}func main() {fmt.Println(triple(4))
}
5.9 宕机(Panic)
有些错误(比如数组越界访问或者解引用空指针)都需要在运行时进行检查。当 Go 语言运行时检测到这些错误,就会发生宕机。
一个典型的宕机发生时,正常的程序执行会终止, goroutine 中所有的延迟函数会执行,然后程序会异常退出并打印一条日志。日志消息包括宕机的值,这往往代表某种错误消息,每一个 goroutine 都会在宕机的时候显示一个函数调用的栈跟踪消息。
宕机发生时,defer 函数会以倒序执行。
func main() {defer printStack()
}func printStack() {var buf [4096]byten := runtime.Stack(buf[:], false)os.Stdout.Write(buf[:n])
}
5.10 恢复
recover 会终止当前的宕机状态,并且返回宕机的值。函数不会从之前宕机的地方继续运行而是正常返回。如果 recover 在没有宕机的情况下调用,没有任何结果,并且返回 nil。
相关文章:
Go语言程序设计-第5章--函数
Go语言程序设计-第5章–函数 5.1 函数声明 每个函数声明都包含一个名字、一个形参列表、一个可选的返回列表以及函数体: func name(parameter-list) (result-list) {body }func add(x int, y int) int { return x y} func sub(x, y int) (z int) {z x - y; return} func f…...
数据被锁?被.mkp 勒索病毒攻击后的拯救行动
导言: 网络安全面临着越来越多的挑战,而.mallox勒索病毒则成为数字威胁中的一股强大势力。它的威胁不仅体现在其高度复杂的加密算法上,还表现在对受感染系统的深度渗透和数据的极大破坏上。以下是.mallox勒索病毒的主要威胁:如不…...
Fine-Tuning Language Models from Human Preferences
Abstract 奖励学习(reward learning)可以将强化学习(RL)应用到由人类判断定义奖励的任务中,通过询问人类问题来构建奖励模型。奖励学习的大部分工作使用了模拟环境,但是关于价值的复杂信息经常是以自然语言的形式表达的。我们相信语言奖励学习是使强化学习在现实世界任务…...
提升数据库性能的关键指南-Oracle AWR报告
文章目录 一、了解AWR报告:数据库性能的仪表盘二、生成AWR报告三、解读AWR报告的关键部分1.报告开头的系统基础信息2.ADDM发现3.负载概览(Load Profile)4.参数文件5.顶级前台等待事件6.SQL 统计信息-顶级SQL7.SGA Advisory AND PAG Advisory 一、了解AWR报告&#x…...
云计算IaaS、PaaS和SaaS之
提供的服务来比较如下两图 示例图 示例图...
解锁大数据世界的钥匙——Hadoop HDFS安装与使用指南
目录 1、前言 2、Hadoop HDFS简介 3、Hadoop HDFS安装与配置 4、Hadoop HDFS使用 5、结语 1、前言 大数据存储与处理是当今数据科学领域中最重要的任务之一。随着互联网的迅速发展和数据量的爆炸性增长,传统的数据存储和处理方式已经无法满足日益增长的需求。…...
写在2023岁末:敏锐地审视量子计算的当下
本周,《IEEE Spectrum》刊登了一篇出色的文章,对量子计算(QC)的近期前景进行了深入探讨。 文章的目的并不是要给量子计算的前景泼冷水,而是要说明量子计算的前景还很遥远,并提醒读者量子计算的用例可能很窄…...
C/C++学习笔记十三 C++中的重载运算符
1、什么是运算符重载? 运算符重载是 C 中的一项功能,使运算符(例如 、- 等)能够处理用户定义的数据类型。这种机制称为编译时多态性,并提供了为不同数据类型定制运算符行为的优点。 例如,我们可以重载“”运…...
Java 实现自动获取法定节假日
一、背景 在实现业务需求的过程中,遇到了需要计算 x 个工作日后的日期需求。由于工作日是每年国务院发布的,调休和休假都没有规律,所以无法使用算法进行计算。 一般的实现方案是自己维护一个工作日和调休的表,或者去爬取国务院发…...
湘潭大学-2023年下学期-c语言-作业0x0a-综合1
A 求最小公倍数 #include<stdio.h>int gcd(int a,int b) {return b>0?gcd(b,a%b):a; }int main() {int a,b;while(~scanf("%d%d",&a,&b)){if(a0&&b0) break;printf("%d\n",a*b/gcd(a,b));}return 0; }记住最大公约数的函数&…...
网络协议-BIO实战和NIO编程
网络通信编程基本常识 在开发过程中,如果类的名字有 Server 或者 ServerSocket 的,表示这个类是给服务端容纳网络 服务用的,如果类的名字只包含 Socket 的,那么表示这是负责具体的网络读写的。 ServerSocket 并不负责具体的网络读…...
Word 将页面方向更改为横向或纵向
文章目录 更改整个文档的方向更改部分页面的方向方法1:方法2: 参考链接 更改整个文档的方向 选择“布局”>“方向”,选择“纵向”或“横向”。 更改部分页面的方向 需要达到下图结果: 方法1: 选:中你要在横向页面…...
关键字:abstract关键字
在 Java 中,abstract是一个关键字,用于修饰类和方法。当一个类被声明为抽象类时,它不能被实例化,只能被其他类继承。同时,抽象类可以包含抽象方法,抽象方法没有方法体,只包含方法的签名…...
从PDF中提取图片
由于工作需要,要从pdf文件中提取出图片保存到本地,项目中就引用到了Apache PDFBox库。 1 什么是Apache PDFBox? Apache PDFBox库,一个用于处理PDF文档的开源Java工具。它允许用户创建全新的PDF文件,操作现有的PDF文档࿰…...
推荐:一个不错的介绍Apache Doris的PPT
原来Apache Doris居然是百度开源出来的,不错。部分节选:完整下载地址网盘: 链接: https://pan.baidu.com/s/18WR70R_f72GxCjh0lykStQ 提取码: umd3 复制这段内容后打开百度网盘手机App,操作更方便哦 --来自百度网盘超级会员v7的分…...
【Python_PySide2学习笔记(二十二)】进度对话框QProgressDialog类的基本用法
进度对话框QProgressDialog类的基本用法 进度对话框QProgressDialog类的基本用法前言一、QProgressDialog 的常用方法1、创建进度对话框2、进度对话框设置窗口标题3、进度对话框隐藏"最大化"、"最小化"、"关闭"4、进度对话框设置是否自动关闭5、…...
使用rust读取usb设备ACR122U的nfc卡片id
rust及其高效和安全著称,而且支持跨平台,所以就想使用这个rust开发一个桌面端程序,来读取nfc设备的nfc卡片的id信息,下面就做一个最简单的入门教程吧,也是我写的第三个rust应用。 当你电脑上安装好了rust环境之后&…...
servlet总结
目录 1.生命周期 2.线程总结 3.配置 4.请求和响应 5.会话管理 6.过滤和监听器 7.处理表单数据 8.与JSP集成 9.异常处理 10.安全性和认证 Servlet是一种基于Java的Web组件,用于处理客户端请求并生成动态Web内容。以下是关于Servlet的一些总结 1.生命周期 …...
Nacos2.1.2改造适配达梦数据库7.0
出于业务需求,现将Nacos改造适配达梦数据库7.0,记录本次改造过程。 文章目录 一、前期准备二、适配流程1、项目初始化2、引入驱动3、源码修改 三、启动测试四、打包测试 一、前期准备 Nacos源码,版本:2.1.2:源码下载…...
TPRI-DMP平台介绍
TPRI-DMP平台介绍 1 TPRI-DMP平台概述 TPRI-DMP为华能集团西安热工院自主产权的工业云PaaS平台,已经过13年的发展和迭代,其具备大规模能源电力行业生产应用软件开发和运行能力。提供TPRI-DMP平台主数据管理、业务系统开发与运行、应用资源管理…...
Android Studio和java语言数字奇门遁甲排盘系统 v1.0源代码使用说明
Android Studio和java语言数字奇门遁甲排盘系统 v1.0源代码使用说明 一、软件简介 Android Studio和java语言数字奇门遁甲排盘系统 v1.0源代码是一款基于 Android Studio 和 Java 开发的国学数术类工具软件的源代码。该系统以传统奇门遁甲理论为基础,结合数字化模型…...
N5110 LCD驱动深度解析:PCD8544嵌入式实战指南
1. N5110 LCD驱动库深度解析:面向嵌入式工程师的PCD8544控制器实战指南Nokia 5110液晶显示屏因其低功耗、高对比度、宽温工作范围及极简硬件接口,长期被嵌入式系统广泛采用。该模块核心控制器为飞利浦(现NXP)PCD8544,一…...
新手避坑指南:用C语言操作txt文件时最容易犯的5个错误(基于EDUcoder实训案例)
C语言文件操作避坑实战:从EDUcoder案例解析5大经典错误 第一次用C语言操作文件时,我盯着屏幕上那个神秘的FILE*指针发了半小时呆——明明代码和教材示例一模一样,为什么运行时总是报"Segmentation fault"?直到深夜调试才…...
欧姆龙CP1H脉冲程序案例及新手入门指南
A1欧姆龙CP1H程序 姆龙标准程序 欧姆龙PLC标准案例模板 本产品适用于新手或者在校生 本程序主要写了欧姆龙CP1H脉冲程序案例, 包含以下: 威纶通触摸屏程序; word详细说明文档 ; 欧姆龙CP1H程序; 里面的文档有详细介绍…...
终极指南:解决object-reflector使用中的20个常见难题
终极指南:解决object-reflector使用中的20个常见难题 【免费下载链接】object-reflector Allows reflection of object attributes, including inherited and non-public ones 项目地址: https://gitcode.com/gh_mirrors/ob/object-reflector object-reflect…...
MDK开发必备:3步搞定bin文件生成与反汇编(附fromelf命令详解)
MDK开发实战:从bin生成到反汇编的深度解析与高效技巧 引言 在嵌入式开发领域,MDK(Microcontroller Development Kit)作为ARM架构下的主流开发环境,其工程配置与构建流程的掌握程度直接影响开发效率。对于刚接触MDK的开…...
C#指针安全实践:在合法范围内高效操作内存的10个关键步骤
你是否曾幻想过"用指针黑入系统"? 当99.9%的开发者误入"指针黑入"陷阱导致系统崩溃/数据泄露,而真正的安全专家正在用100%合法的内存操作提升300%系统性能——本文将用100%可运行的深度安全代码,从.NET内存模型底层到合法…...
Agent智能体架构 第二章 单智能体架构
单智能体架构 (Single Agent) 这是最简单的形式,指代的是一个智能体独立完成所有任务。代表:AutoGPT、BabyAGI 的早期版本。优点:上下文一致性强,没有协作开销。缺点:能力受限于单一模型的上下文窗口,难以处…...
打卡信奥刷题(2993)用C++实现信奥题 P6121 [USACO16OPEN] Closing the Farm G
P6121 [USACO16OPEN] Closing the Farm G 题目背景 本题和 银组同名题目 在题意上一致,唯一的不同是数据范围。 题目描述 FJ 和他的奶牛们正在计划离开小镇做一次长的旅行,同时 FJ 想临时地关掉他的农场以节省一些金钱。 这个农场一共有被用 MMM 条…...
从零到一:PointNet实战全流程解析与避坑指南
1. PointNet入门:为什么选择这个框架? 第一次接触3D点云处理时,我被各种复杂的算法搞得头晕眼花,直到发现了PointNet这个优雅的解决方案。与传统的体素化或投影方法不同,PointNet直接处理原始点云数据,这种…...
