当前播放: 14 | 可变参数和defer
00:00 / 00:00
高清
  • 高清
1.0x
  • 2.0x
  • 1.5x
  • 1.25x
  • 1.0x
  • 0.5x
网页全屏
全屏
00:00
付费课程,可试看
课程目录
第一章:Go语言简介 (4讲)
01 | Go语言课程介绍
免费
02 | 内容综述
免费
03 | Go语言简介:历史背景、发展现状及语言特性
免费
04 | 编写第一个Go程序
免费
第二章:基本程序结构 (4讲)
05 | 变量、常量以及与其他语言的差异
免费
06 | 数据类型
07 | 运算符
08 | 条件和循环
第三章:常用集合 (3讲)
09 | 数组和切片
10 | Map声明、元素访问及遍历
11 | Map与工厂模式,在Go语言中实现Set
第四章:字符串 (1讲)
12 | 字符串
第五章:函数 (2讲)
13 | Go语言的函数
14 | 可变参数和defer
第六章:面向对象编程 (4讲)
15 | 行为的定义和实现
16 | Go语言的相关接口
17 | 扩展与复用
18 | 不一样的接口类型,一样的多态
第七章:编写好的错误处理 (2讲)
19 | 编写好的错误处理
20 | panic和recover
第八章:包和依赖管理 (2讲)
21 | 构建可复用的模块(包)
22 | 依赖管理
第九章:并发编程 (7讲)
23 | 协程机制
24 | 共享内存并发机制
25 | CSP并发机制
26 | 多路选择和超时
27 | channel的关闭和广播
28 | 任务的取消
29 | Context与任务取消
第十章:典型并发任务 (5讲)
30 | 只运行一次
31 | 仅需任意任务完成
32 | 所有任务完成
33 | 对象池
34 | sync.pool对象缓存
第十一章:测试 (3讲)
35 | 单元测试
36 | Benchmark
37 | BDD
第十二章:反射和Unsafe (3讲)
38 | 反射编程
39 | 万能程序
40 | 不安全编程
第十三章:常见架构模式的实现 (2讲)
41 | 实现pipe-filter framework
42 | 实现micro-kernel framework
第十四章:常见任务 (4讲)
43 | 内置JSON解析
44 | easyjson
45 | HTTP服务
46 | 构建RESTful服务
第十五章:性能调优 (4讲)
47 | 性能分析工具
48 | 性能调优示例
49 | 别让性能被锁住
50 | GC友好的代码
第十六章:高可用性服务设计 (5讲)
51 | 高效字符串连接
52 | 面向错误的设计
53 | 面向恢复的设计
54 | Chaos Engineering
55 | 结课测试&结束语
14 | 可变参数和defer

14 | 可变参数和defer

蔡超
Mobvista技术副总裁兼首席架构师,前亚马逊(中国)首席软件架构师
全集8604
新人首单 ¥29.9 原价 ¥129
9
本节摘要
展开
登录 后留言

精选留言(14)

  • Flygar
    作为defer的补充:
    1. Defer 被用来确保一个函数调用在程序执行结束前执行。同样用来执行一些清理工作。defer 用在像其他语言中的 ensure 和 finally用到的地方。
    2. 关键字 defer 允许我们推迟到函数返回之前(或任意位置执行 return 语句之后)一刻才执行某个语句或函数(为什么要在返回之后才执行这些语句?因为 return 语句同样可以包含一些操作,而不是单纯地返回某个值)。
    3. 推迟的函数调用会被压入一个栈中。当外层函数返回时,被推迟的函数会按照后进先出的顺序调用。

    //当有多个 defer 行为被注册时,它们会以逆序执行(类似栈,后进先出)
    func fdefer() {
    for i := 0; i < 5; i++ {
    defer fmt.Printf("%d\n", i)
    }
    }

    //变量 ret 的值为 2,因为 ret++ 是在执行 return 1 语句后发生的
    func fdefer2() (ret int) {
    defer func() {
    ret++
    }()
    return 1
    }
    2019-03-14
    1
    12
  • Steven Fung
    1,从打印来看,panic在defer之后,为什么还说defer是在panic之后执行?
    2,panic应该也有非自己调用的,这种是怎么保证defer在panic之后执行?

    作者回复: 你尝试都用fmt.Println输出

    2019-03-19
    2
  • 王宇
    可变长参数,看到了PHP的影子…
    2020-02-10
    1
    1
  • 周棋洛
    老师,是否可以这么理解,defer像是函数的析构造?

    作者回复: 不太一样,析构函数的执行时间是和对象的生命周期相关,而defer可以用在非对象的场景

    2019-06-19
    1
  • 梧桐
    初学语言,用中文解释了一下,方便理解
    ```
    // defer「延缓」碰到这个函数,先延缓不动。在该函数最后返回前,才执行defer声明的函数, 一般用来释放执行函数后的资源和锁
    // panic「惊恐」让人感到惊恐的错误 ⚠️:即便出现严重错误,defer 依然可以执行,用于释放资源

    func Clear(par1 int) {
    fmt.Println("3. Clear resources!", par1)
    }

    func FuncDefer() {
    fmt.Println("2. 被 defer 的函数里面还有被 defer 的化,会被先执行!")
    defer Clear(2)
    }

    func TestDefer(t *testing.T) {
    defer Clear(1)
    defer FuncDefer()
    fmt.Println("1. Start")
    panic("err")
    }

    ```
    2019-06-05
    1
    1
  • Amorfati
    请教老师,我在写defer的这个例子的时候
    func TestDefer(t *testing.T) {
    defer Clear()
    fmt.Println("start")
    t.Log("after start")
    panic("Fatal Err")
    }
    after start 这个并没有显示出来,能麻烦帮忙解答一下为什么没有显示出来么

    作者回复: 改成用fmt.Println(“after start”)

    2019-03-17
    1
  • 郭星
    关于 defer func()func() 的问题
    func GetFunc() func() {
       fmt.Print("[outside]")
       return func() {
          fmt.Print("[inside]")
       }
    }


    // 输出结果为 [outside][here][inside]
    func TestDeferGetFunc(t *testing.T) {
       defer GetFunc()()
       fmt.Print("[here]")
    }
    为什么会先执行GetFunc() 输出[outside]

    作者回复: 这个在我的公众号里,讨论过这个问题。 defer延迟执行只是最后一层调用的延迟

    2020-07-28
  • Cs
    作为defer的补充:
    1. Defer 被用来确保一个函数调用在程序执行结束前执行。同样用来执行一些清理工作。defer 用在像其他语言中的 ensure 和 finally用到的地方。
    2. 关键字 defer 允许我们推迟到函数返回之前(或任意位置执行 return 语句之后)一刻才执行某个语句或函数(为什么要在返回之后才执行这些语句?因为 return 语句同样可以包含一些操作,而不是单纯地返回某个值)。
    3. 推迟的函数调用会被压入一个栈中。当外层函数返回时,被推迟的函数会按照后进先出的顺序调用。

    //当有多个 defer 行为被注册时,它们会以逆序执行(类似栈,后进先出)
    func fdefer() {
    for i := 0; i < 5; i++ {
    defer fmt.Printf("%d\n", i)
    }
    }

    //变量 ret 的值为 2,因为 ret++ 是在执行 return 1 语句后发生的
    func fdefer2() (ret int) {
    defer func() {
    ret++
    }()
    return 1
    }
    2020-05-24
  • 小虎
    Defer延迟执行,相当于java里面的finally,无论是否程序中断都会执行,一般用于资源清理。
    2020-03-26
  • o9
    稍微解释了一下:
    ```go
    // 变量名称(innerFunc) 类型(函数类型) 返回值(函数类型)
    func timeSpent(innerFunc func(op int) int) func(op int) int {
    // 返回一个函数, 这个函数又一个整形入参 n
    return func(n int) int {
    // 函数开始执行时间
    start := time.Now()
    ret := innerFunc(n)
    // 打印计算出的 函数执行花费的时间
    fmt.Println("Time Spent: ", time.Since(start).Seconds())

    // 返回 传入函数(也就是 innerFunc)的返回值
    return ret
    }
    }

    // 定义一个符合 timeSpent 入参的函数, 供后面测试
    func slowFlow(n int) int {
    time.Sleep(time.Second * 1)
    return n
    }

    func TestFn(t *testing.T) {
    a, b := returnMultiValues()
    t.Log(a, b)

    // 调用 timeSpent 函数, 传入 slowFlow 函数
    ret := timeSpent(slowFlow)
    // 打印返回值
    t.Log(ret(10))
    }
    ```
    2020-01-08
  • WING
    老师,defer的声明是要写在程序最前面吗?我试了一下,如果defer 申明放在panic后面,是不会执行的,这根java的finally有点区别。
    fmt.Println("Start...")
    panic("Fatal error.")
    defer Clear()
    我这样写,Clear()不会被执行。

    作者回复: 不用写在最前面,就例子而言要写在你的panic前面。但是在实际程序中由于我们不知道错误/panic会在哪里发生所以把defer写在最前面会更简单保险。

    2019-12-08
  • 风吹过
    好别扭,跟我读:滴fer,不是跌fer [dɪˈfɜː(r)]

    作者回复: 这个是短音I,不发“滴”。找个发音软件

    2019-06-03
    1
  • 张sir
    可变长参数应该是切片吧
    2019-05-05
  • 万卷苏
    请教老师,我在写defer的这个例子的时候
    func TestDefer(t *testing.T) {
        defer Clear()
        fmt.Println("start")
        t.Log("after start")
        panic("Fatal Err")
    }
    after start 这个并没有显示出来,能麻烦帮忙解答一下为什么没有显示出来么

    这个并不是after start没显示出来, 而是在不同的地方显示了, t.Log输出的内容, 和panic输出的内容, 在同一个地方, 可能这两个属于同一类方法吧

    作者回复: 这个是用了不同的输出方式的原因,都用fmt.Println输出试试

    2019-03-31
收起评论
看过的人还看
Go语言核心36讲

郝林  《Go并发编程实战》作者,前轻松筹大数据负责人

55讲 | 26292 人已学习

新人首单 ¥19.9 原价 ¥99
数据结构与算法之美

王争  前Google工程师

80讲 | 89754 人已学习

新人首单 ¥29.9 原价 ¥129
MySQL实战45讲

林晓斌  网名丁奇,前阿里资深技术专家

49讲 | 58432 人已学习

新人首单 ¥29.9 原价 ¥129
趣谈网络协议

刘超  网易研究院云计算技术部首席架构师

51讲 | 45923 人已学习

新人首单 ¥19.9 原价 ¥99