go是函数式编程语言吗?

时间:2022-12-31 20:28:23
2012-05-19 翻译自这里, 对原文有所扩展, 也有所删减.
 
go是函数式编程语言吗?
不是, 当然不是.
 
那么, go提供函数吗?
是的, 当然, 大多数编程语言都提供函数, go也不例外. 不相信吗? 我会用代码让你闭嘴:
func SayHello() {
fmt.Println(
"Hello")
}

看见了吧. go使用关键字func定义函数, 并在函数体中编写函数逻辑. 

 
go函数可以接受参数吗?
嗯, 我又看到一个白痴的问题, 呵呵. 哦, 我懂了, 也许是我的SayHello函数给大家造成了错觉, 我会改造我的代码:
func SayHelloToSomeone(name string) {
fmt.Println(
"Hello " + name + ".")
}

函数SayHelloToSomeone接受一个string类型的参数name.

 
go函数是否可以有返回值?
是的, 是的, 是的! 就像数学意义上的函数一样, go函数可以返回给调用者一些东西. 为了演示这一点, 我将重新编写一个函数:
func GetGreeting (name string) string {
greeting :
= "Hello " + name + "."
return greeting
}
// test
greeting := GetGreeting("Bob")
fmt.Println(greeting)
//outputs "Hello Bob."

当然这没有什么, 其他语言也可以做到. 但是, 准备好接受惊喜了吗? 

go函数的返回值与其他类C语言有些不同, 比如, 你可以为返回值指定名称. 这带来至少2个好处:
1. 不需要在函数体中为返回值定义变量.
2. 无需在return语句后加上返回值. go会自动将返回值加上.
func GetGreeting (name string) (greeting string) {
greeting
= "Hello " + name + "."
return
}

GetGreeting函数为其返回值指定了名称: greeting. 在函数体中就可以直接使用greeting变量了, greeting其实就相当于函数中定义的一个局部变量. 而且如你所见, GetGreeting函数的return语句后面没有加上返回值, 因为go会自动将greeting变量的值返回给调用者.

 
go函数可以返回多个值吗?
这, 这, 你是异想天开吗? 不过强大的go函数能够做到这一点, 哈哈. 要知道可以有多个返回值的函数可以避免很多丑陋的代码, 下面是示例:
type Stack struct {
pos
int
data [
10]int
}
func (s
*Stack) Pop() (value int, ok bool) {
if s.pos > 0 {
s.pos
--
ok
= true
value
= s.data[s.pos]
return
}
ok
= false
return
}

代码中首先定义了Stack类型, 并提供了Pop函数. Pop函数返回2个值: value和ok, 分别代表pop的值和pop操作是否成功.

 
go函数可以接受一个函数作为参数吗?
嗯, 我想你终于开始集中注意力了. 
如果你是一个医生, 你是否会对每次都需要向你的病人SayHello感到厌烦? 没关系, go可以帮助你. 首先需要定义新的数据结构:
type TormentList struct {
patients []
string
}
// 将[]string(string数组)包装成TormentList类型的指针
func NewTormentList(people []string) *TormentList {
return &TormentList{people}
}

接下来, 让我们悄悄给TormentList类型增加Map方法:

func (g *TormentList) Map(f func(string)) {
// 遍历g.patients, 为其每个value调用f方法
for _, val := range(g.patients) {
f(val)
}
}

Map方法接受f函数作为其输入参数, f函数接受一个string类型的值. 遍历TormentList的病人, 并为每个病人调用f函数.

现在已经做好了一切准备, 剩下的就是测试了:
patients := []string{"Anand", "David", "Ivan", "JoJo", "Jin", "Mon", "Peter", "Sachin"}
gl :
= NewTormentList(patients)
// 还记得上面定义的SayHelloToSomeone函数吧?
gl.Map(SayHelloToSomeone)

/*
outputs the following:

Hello Anand.
Hello David.
Hello Ivan.
Hello JoJo.
Hello Jin.
Hello Mon.
Hello Peter.
Hello Sachin.
*/ 
 
go函数的返回值可以是函数吗?
让我们先考虑一个现实问题: 假如你拥有一份吃过寿司的人的清单, 你是否能够根据人名确定他是否在清单上? 这是个很简单的问题, 你只需遍历清单. 嗯, 如果你go的功底很弱, 不知道怎么遍历清单那怎么办? 没关系, 我会给你提供一个刷选器:
func Screen(patients []string) func(string) bool {
// 定义匿名函数并返回
return func(name string) bool {
for _, soul := range patients {
if soul == name {
return true
}
}
return false
}
}
Screen方法会将刷选的函数返回给调用方, 这样你就可以不用懂怎么去遍历清单了, 你只需调用我返回给你的函数就可以:
// 吃过寿司的人的清单
those_who_bought_sushi := []string{"Anand", "JoJo", "Jin", "Mon", "Peter", "Sachin"}
// 得到刷选器函数
bought_sushi := Screen(those_who_bought_sushi)
// 调用刷选器函数就可以知道某人是否在清单上
fmt.Println(bought_sushi("Anand")) // true
fmt.Println(bought_sushi("Alex")) // false