博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Go语言学习笔记(八)
阅读量:4691 次
发布时间:2019-06-09

本文共 3341 字,大约阅读时间需要 11 分钟。

函数进阶

函数作用域

1.全局变量,在程序整个生命周期都是有效的

2.局部变量:函数内定义/语句块内定义

变量可变性

包内任何变量或者函数都是能访问的。包外的话,首字母大写是可导出的能够被其他包访问或调用。小写表示是私有的,不能被外部的包访问。同一个包内大小写的全局变量或者函数是能被直接调用的。

跨包导入和调用:
 
1: 导入:
2: import (
3: 	github.com\apple_school\listen1\function\calc"
4: )
5: 调用:
6: calc.Add()
7: calc.A
匿名函数
 
1: func add(a, b int) int {
2: 	return a + b
3: }
4: func test(){
5: 	f1 := add
6: 	fmt.Printf("type of f1=%T\n", f1)
7: }
8: func main() {
9: 	test()
10: }
 
1: func test_1() {
2: 	f1 := func(a, b int) int {
3: 		return a + b
4: 	}
5: 	fmt.Printf("type of f1=%T\n", f1)
6: }

defer中使用匿名函数,注意下面两种区别

 
1: func test_2() {
2: 	var i int = 0
3: 	defer fmt.Printf("i=%d\n", i) // defer里i=0传进去了
4: 	i = 100
5: 	return
6: }
7: >>> i = 0
8:
9: func test_3() {
10: 	var i int = 0
11: 	defer func() {
12: 		fmt.Printf("defer i=%d\n", i)
13: 	} ()
14: 	i = 100
15: 	return
16: }
17: >>> defer i=100

函数作为参数

 
1: func add(a, b int) int {
2: 	return a + b
3: }
4:
5: func test_c(a, b int32) int32 {
6: 	return a * b
7: }
8:
9: func sub(a, b int) int {
10: 	return a - b
11: }
12:
13: func calc(a, b int,op func(int, int) int) int {
14: 	return op(a, b)
15: }
16:
17: func test_4() {
18: 	sum := calc(100, 300, add)
19: 	sub := calc(100, 300, sub)
20: 	su := calc(100, 300, test_c)
21: 	fmt.Printf("sum=%d sub=%d\n", sum, sub)
22: }
23:

闭包

即函数定义和函数表达式位于另一个函数的函数体内(嵌套函数)。而且,这些内部函数可以访问它们所在的外部函数中声明的所有局部变量、参数。当其中一个这样的内部函数在包含它们的外部函数之外被调用时,就会形成闭包。

也就是说,内部函数会在外部函数返回后被执行。而当这个内部函数执行时,它仍然必需访问其外部函数的局部变量、参数以及其他内部函数。这些局部变量、参数和函数声明(最初时)的值是外部函数返回时的值,但也会受到内部函数的影响。

例子1:

 
1: package main
2: import "fmt"
3:
4: func add(base int) func(int) int {
5: 	return func(i int) int {
6: 		base += i
7: 		return base
8: 	}
9: }
10: func main() {
11: 	tmp1:= add(10)
12: 	fmt.Println(tmp1(1), tmp1(2), tmp1(3))
13: 	tmp2:= add(100)
14: 	fmt.Println(tmp2(1), tmp2(2), tmp2(3))
15: }
16: >>> 11 13 16
17: 101 103 106

例子2:

 
1: package main
2: import (
3: 	"fmt"
4: 	"strings"
5: )
6:
7: func makeSuffix(suffix string) func(string) string {
8: 	return func(name string) string {
9: 		if !strings.HasSuffix(name, suffix) {
10: 			return name + suffix
11: 		}
12: 		return name
13: 	}
14: }
15: func main() {
16: 	func1 := makeSuffixFunc(".bmp")
17: 	func2 := makeSuffixFunc(".jpg")
18: 	fmt.Println(func1("test"))
19: 	fmt.Println(func2("test"))
20: }
21:
22: >>> test.bmp
23: test.jpg

用python3来写就是这样(比较容易理解)

 
1: def makeSuffix(suffix):
2:
3: 	def func(name):
4: 		if  not name.endswith(suffix):
5: 			return name + suffix
6: 	return func
7: func = makeSuffix(".bmg")
8: print(func("test"))

例子3:

 
1: func calc(base int) (func(int) int, func(int) int) {
2: 	add:= func(i int) int {
3: 		base += i
4: 		return base
5: 	}
6: 	sub:= func(i int) int {
7: 		base -= i
8: 		return base
9: 	}
10: 	return add, sub
11: }
12:
13: func main() {
14: 	f1, f2 := calc(10)
15: 	fmt.Println(f1(1), f2(2))
16: 	fmt.Println(f1(3), f2(4))
17: 	fmt.Println(f1(5), f2(6))
18: 	fmt.Println(f1(7), f2(8))
19: }
20:
21: >>> 11 9
22: 12 8
23: 13 7
24: 14 6

闭包的意义

返回的函数对象,不仅仅是一个函数对象,在该函数外还包裹了一层作用域,这使得,该函数无论在何处调用,优先使用自己外层包裹的作用域

在多线程下的闭包的副作用:(了解就行了)
 
1: package main
2:
3: import (
4: 	"fmt"
5: 	"time"
6: )
7: func main() {
8: 	for i:=0;i<5;i++ {
9: 		go func(){
10: 			fmt.Println(i)
11: 		} ()
12: 	}
13: 	time.Sleep(time.Second*1)
14: }
几次的输出结果你会发现不但数值不是固定的,而且居然还会打印出
5。本来循环在i=5是就停止了,但是协程搞了鬼,这个后面再讲。

转载于:https://www.cnblogs.com/haoqirui/p/10115103.html

你可能感兴趣的文章
Mybatis逆向工程配置文件详细介绍(转)
查看>>
String类的深入学习与理解
查看>>
不把DB放进容器的理由
查看>>
OnePage收集
查看>>
Java parseInt()方法
查看>>
yahoo的30条优化规则
查看>>
[CCF2015.09]题解
查看>>
[NYIST15]括号匹配(二)(区间dp)
查看>>
json_value.cpp : fatal error C1083: 无法打开编译器生成的文件:No such file or directory
查看>>
洛谷 P1101 单词方阵
查看>>
Swift DispatchQueue
查看>>
C#和JAVA 访问修饰符
查看>>
小甲鱼OD学习第1讲
查看>>
HDU-1085 Holding Bin-Laden Captive-母函数
查看>>
php提示undefined index的几种解决方法
查看>>
LRJ
查看>>
Struts2环境搭建
查看>>
Linux: Check version info
查看>>
Javascript-正则表达式-开发中的使用.
查看>>
stl学习之测试stlen,cout等的运行速度
查看>>