package main

import "fmt"

// function add
func add(a, b int) int {
return a + b
}

// 1
func testFunc1() {

// function "add" to var "f1"
// then "f1" is a function
f1 := add

// type of f1 = func(int int) int
fmt.Printf("type of f1 = %T\n", f1)

// call function "f1"
// params are 2 and 5
sum := f1(2, 5)

// sum = 7
fmt.Printf("sum = %d\n", sum)

}

// 2
func testFunc2() {

// anonymous function to var "f1", then "f1" is a function
f1 := func(a, b int) int {
    return a + b
}

// type of f1 = function(int, int) int
fmt.Printf("type of f1 = %T\n", f1)

// call function f1, params are 2 and 5
sum := f1(2, 5)

// sum = 7
fmt.Printf("sum = %d\n", sum)

}

// 3
func testFunc3() {
var i = 0

// the statement after "defer" will be pushed into stack first
// so the value of var "i" will be "0"
// defer i = 0
defer fmt.Printf("defer i = %d\n", i)

i = 100

// i = 100
fmt.Printf("i = %d\n", i)

return

}

// 4
func testFunc4() {
var i = 0

// the anonymous function after "defer" will be pushed into stack first
// but at this time, the statement in function will not be pushed into stack
// so at this time the value of var "i" is not specific
// the value of var
// at the end the value of var "i" is 100
defer func() {
    fmt.Printf("defer i = %d\n", i)
}()

i = 100

// i = 100
fmt.Printf("i = %d\n", i)

return

}

func main() {
//testFunc1()
//testFunc2()
//testFunc3()
testFunc4()
}