golang学习笔记(defer基础知识)

什么是defer

defer语句用于golang程序中延迟函数的调用, 每次defer都会把一个函数压入栈中, 函数返回前再把延迟的函数取出并执行。

为了方便描述, 我们把创建defer的函数称为主函数, defer语句后面的函数称为延迟函数。延迟函数可能有输入参数, 这些参数可能来源于定义defer的函数, 延迟函数也可能引用主函数用于返回的变量, 也就是说延迟函数可能会影响主函数的一些行为。

defer的规则

规则一:延迟函数的参数在defer语句出现时就已经确定

package main

import "fmt"

func main() {
	deferFuncParameter()
}

func deferFuncParameter() {
	var aInt = 1
	defer fmt.Println(aInt)
	aInt = 2
	return
}

结果:
在这里插入图片描述
代码说明: 函数deferFuncParameter()定义一个整型变量并初始化为1,然后使用defer语句打印出变量值, 最后修改变量值为2。
参考答案: 输出1。 延迟函数fmt.Println(aInt)的参数在defer语句出现时就已经确定了, 后面修改的aInt变量实际上是拷贝了一份。所以无论后面如何修改aInt变量都不会影响延迟函数的执行。
注意: 对于指针类型参数, 规则仍然适用, 只不过延迟函数的参数是一个地址值, 这种情况下,defer后面的语句对变量的修改可能会影响延迟函数。

package main

import "fmt"

func main() {
	deferArray()
}

func printArray(array *[3]int)  {
	for i := range array {
		fmt.Println(array[i])
	}
}

func deferArray()  {
	var aArray = [3]int{1, 2, 3}
	defer printArray(&aArray)
	aArray[0] = 10
	return
}

结果:
在这里插入图片描述
函数说明: 函数deferFuncParameter()定义一个数组, 通过defer延迟函数printArray()的调用, 最后修改数组第一个元素。 printArray()函数接受数组的指针并把数组全部打印出来。
参考答案: 输出10、 2、 3三个值。 延迟函数printArray()的参数在defer语句出现时就已经确定了, 即数组的地址, 由于延迟函数执行时机是在return语句之前, 所以对数组的最终修改值会被打印出来。

规则二:defer延迟函数执行按后进先出顺序执行, 即先出现的defer最后执行

定义defer类似于入栈操作, 执行defer类似于出栈操作。

设计defer的初衷是简化函数返回时资源清理的动作, 资源往往有依赖顺序, 比如先申请A资源, 再跟据A资源申请B资源, 跟据B资源申请C资源, 即申请顺序是:A—>B—>C, 释放时往往又要反向进行。 这就是把deffer设计成FIFO的原因。每申请到一个用完需要释放的资源时, 立即定义一个defer来释放资源是个很好的习惯。

规则三: 延迟函数可能操作主函数的具名返回值

定义defer的函数, 即主函数可能有返回值, 返回值有没有名字没有关系, defer所作用的函数, 即延迟函数可能会影响到返回值。

package main

import "fmt"

func main() {
	fmt.Println(test())
}
func test() (res int) {
	a := 1
	defer func() {
		res ++
	}()
	return a
}

结果
在这里插入图片描述
函数说明: 函数拥有一个具名返回值result, 函数内部声明一个变量a, defer指定一个延迟函数, 最后返回变量a。延迟函数中递增res。
参考答案: 函数输出2。 函数的return语句并不是原子的, 实际执行分为设置返回值—>ret, defer语句实际执行在返回前, 即拥有defer的函数返回过程是: 设置返回值—>执行defer—>res。 所以return语句先把res设置为a的值, 即1, defer语句中又把res递增1, 所以最终返回2。
return 返回值解析
该函数的return语句可以拆分成下面两行:

result = i
return

而延迟函数的执行正是在return之前, 即加入defer后的执行过程如下:

result = i
result++
return

一个主函数拥有一个匿名的返回值, 返回时使用字面值, 比如返回”1”、 ”2”、 ”Hello”这样的值, 这种情况下defer语句是无法操作返回值的。
另外返回值是匿名类型的值,这种情况下defer语句可以引用到返回值, 但不会改变返回值。

package main

import "fmt"

func main() {
	fmt.Println(test())
}
func test() int {
	a := 1
	defer func() {
		a ++
	}()
	return a
}

func printArray(array *[3]int)  {
	for i := range array {
		fmt.Println(array[i])
	}
}

结果:
在这里插入图片描述
上面的函数, 返回一个局部变量, 同时defer函数也会操作这个局部变量。 对于匿名返回值来说, 可以假定仍然有一个变量存储返回值, 假定返回值变量为”anony”, 上面的返回语句可以拆分成以下过程:

anony = a
 a++
return

由于a是整型, 会将值拷贝给anony, 所以defer语句中修改i值, 对函数返回值不造成影响。

总结

  1. defer定义的延迟函数参数在defer语句出时就已经确定下来了
  2. defer定义顺序与实际执行顺序相反
  3. return不是原子操作,执行过程是: 保存返回值(若有)—>执行defer( 若有) —>执行ret跳转
  4. 申请资源后立即使用defer关闭资源是好习惯

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/571374.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【Burpsuite靶场】XSS专题精讲

【个人】:NEUQ大一学生 【专业】:通信工程 (Communication Engineering) 【个人方向】:网安、开发双管齐下 【座右铭】:真正的英雄主义,就是看清生活的真相后依然热爱生活 -- 罗曼.罗兰 一、认识XSS(跨站脚本攻击&…

fatal: unable to access ‘https://github.com/alibaba/flutter_boost.git/

Git error. Command: git fetch stdout: stderr: fatal: unable to access ‘https://github.com/alibaba/flutter_boost.git/’: Failed to connect to github.com port 443 after 75005 ms: Couldn’t connect to server exit code: 128 GitHub (国际型)代码 分发平台/托管平…

梯度下降法总是在同一点收敛吗?

梯度下降法总是在同一点收敛吗? 梯度下降法并不总是在同一点收敛。梯度下降法的收敛取决于多个因素,包括初始参数的选择、学习率的设置、损失函数的形状等。 以下是一些影响梯度下降法收敛行为的关键因素: 1.初始参数: 初始参数…

Json-server 模拟后端接口

json-server,模拟rest接口,自动生成增删改查接口。(官网地址:json-server - npm) 使用方法: 1. 安装json-server,npm i json-server -g 2. 创建json文件,文件中存储list数据,db.json {"…

图像超分辨率技术在AI去衣中的应用探索

在数字图像处理领域,图像超分辨率(Super-Resolution, SR)技术一直是研究的热点之一。该技术旨在从低分辨率的图像中恢复出高分辨率的图像,以提供更清晰、更丰富的细节信息。近年来,随着人工智能(AI&#xf…

<计算机网络自顶向下> 路由器组成

路由器结构概况 路由:运行路由选择算法/协议(RIP, OSPF, BGP)生成路由表转发:从输入到输出链路交换数据包-根据路由表进行分组的转发中间的fabric是用来接收输入的分组交给输出端口的,完成局部的转发(根据…

free 命令示例

目录 ⛳️推荐 前言 Linux 中如何使用 free 命令 1、以人类可读的形式显示信息 2、连续显示统计数据 3、定义显示统计数据的次数 4、指定输出数据类型 5、获取物理内存和交换内存的总和 总结 ⛳️推荐 前些天发现了一个巨牛的人工智能学习网站,通俗易懂&am…

掌握注册唤起应用的秘诀,Xinstall助你提升用户体验

在移动互联网时代,App已经成为我们日常生活中不可或缺的一部分。然而,随着App数量的激增,如何让自己的App在激烈的市场竞争中脱颖而出,成为开发者们关注的焦点。其中,注册唤起应用作为提升用户体验和转化率的关键环节&…

4- 24

day02 1.100个英语单词 2.vp div3 不过有点小悲惨,第一题正常的直接看出来答案。第二题其实是map模拟,一直没有读懂题目的意思,题目给的序列是打乱的。找出最小的,讲原来的序列补全,如果mp中没有这个数字,…

Linux网络-DHCP原理与配置

目录 一.DHCP工作原理 1.了解DHCP服务 1.1.使用DHCP的好处 1.2.DHCP的分配方式 2.DHCP的租约过程 2.1.DHCP工作原理 2.2.DHCP交互过程 二.DHCP服务器的配置 1.关闭防火墙 2.检查并且安装DHCP有关软件包 3.查看系统的配置文件 3.1.设置参数 4.修改网络 4.1.修改虚…

python高级进阶(一)[str字符串、set集合、dict字典]

目录 一、str字符串 1. 字符串的概念 2.字符串的特点 3. 定义字符串 4. 获取字符串中的某个元素 5. 遍历字符串 6. 字符串的常用方法 6.1 判断 6.2 转换 6.3 查找 6.4 切割 6.5 去空白 7. 小案例【用户名和密码合法校验】 8. 常用方法中 isdecimal() 和 isdigi…

前端开发攻略---封装calendar日历组件,实现日期多选。可根据您的需求任意调整,可玩性强。

1、演示 2、简介 1、该日历组件是纯手搓出来的,没依赖任何组件库,因此您可以随意又轻松的改变代码,以实现您的需求。 2、代码清爽干净,逻辑精妙,您可以好好品尝。 3、好戏开场。 3、代码(Vue3写法&#xff…

分布式与一致性协议之CAP(一)

CAP理论 概述。 在开发分布式系统的时候,会遇到一个非常棘手的问题,那就是如何根据业务特点,为系统设计合适的分区容错一致性模型,以实现集群能力。这个问题棘手在当发生分区错误时,应该如何保障系统稳定运行而不影响…

C++多态(个人笔记)

C多态 1.多态的定义以及实现1.1多态的构成条件1.2虚函数1.3虚函数的重写1.4override和final1.5函数重载,覆盖(重写),隐藏(重定义)区别 2.抽象类2.1接口继承和实现继承的区别 3.多态原理3.1虚函数表3.2多态的…

SpringBoot整合七牛云实现图片的上传管理

唠嗑部分 各位小伙伴大家好,我是全栈小白,今天我们来分享一下SpringBoot如何整合七牛云存储实现图片的上传与存储 首先我们来说说图片存储,在项目中图片几乎是必不可少的,那么大家会选择怎样存储呢,当然有几种方案 …

软件游戏缺失d3dcompiler_43.dll怎么修复?分享多种靠谱的解决方法

在我们日常频繁地操作和使用电脑的过程中,时常会遇到一些突发的技术问题。其中一种常见的情况是,在尝试启动或运行某个应用程序时,系统会弹出一个错误提示窗口,明确指出当前电脑环境中缺少了一个至关重要的动态链接库文件——d3dc…

算法学习笔记Day9——动态规划初探

一、介绍 本文解决几个问题:动态规划是什么?解决动态规划问题有什么技巧?如何学习动态规划? 1. 动态规划问题的一般形式就是求最值。动态规划其实是运筹学的一种最优化方法,只不过在计算机问题上应用比较多&#xff…

STM32cubemx和HAL库的使用入门--点亮一颗LED

一:流程介绍 (1)环境搭建 1 :stm32cubemx安装 2 :stm32xxFW安装 3 :MDK5安装 4 :生成MDK版本project (2)stm32cubemx创建工程,选择芯片型…

删除链表的倒数第n个节点的最优算法实现

给你一个链表&#xff0c;删除链表的倒数第 n 个结点&#xff0c;并且返回链表的头结点。 提示&#xff1a; 链表中结点的数目为 sz 1 < sz < 300 < Node.val < 1001 < n < sz 你能尝试使用一趟扫描实现吗&#xff1f; 具体实现 要删除链表的倒数第 n 个…

OpenHarmony语言基础类库【@ohos.url (URL字符串解析)】

说明&#xff1a; 本模块首批接口从API version 7开始支持。后续版本的新增接口&#xff0c;采用上角标单独标记接口的起始版本。 导入模块 import Url from ohos.url URLParams9 URLParams接口定义了一些处理URL查询字符串的实用方法。 constructor9 constructor(init?…