Go学习:函数式编程

编程入门 行业动态 更新时间:2024-10-12 01:28:29

Go学习:<a href=https://www.elefans.com/category/jswz/34/1771370.html style=函数式编程"/>

Go学习:函数式编程

函数式编程VS函数指针

  • 函数是一等公民:参数,变量,返回值都可以是函数
  • 高阶函数:函数的参数还可以是函数
  • 函数->闭包

"正统"函数式编程

  • 不可变性:不能有状态,只有常量和函数
  • 函数只能有一个参数
  • go语言是一个通用的语言,它不会再正统函数式编程做过多的文章
package mainimport "fmt"type iAdder func(int) (int, iAdder)func adderi(base int) iAdder {return func(i int) (int, iAdder) {return base + i, adderi(base + i)}
}func main() {a := adderi(0)for i := 0; i < 10; i++ {var s ints, a = a(i)// 0+1+...+0=0// 0+1+...+1=1// 0+1+...+2=3// 0+1+...+3=6// 0+1+...+4=10// 0+1+...+5=15// 0+1+...+6=21// 0+1+...+7=28// 0+1+...+8=36// 0+1+...+9=45fmt.Printf("0+1+...+%d=%d\n", i, s)}}

闭包

闭包可以理解成 “定义再一个函数内部的函数”.再本质上闭包是将函数内部和函数外部连接起来的桥梁.或者说是函数和其引用环境的组合体

package mainimport "fmt"func adder() func(int) int {sum := 0return func(i int) int {sum += ireturn sum}
}func main() {a := adder()for i := 0; i < 10; i++ {// 0+1+...+0=0// 0+1+...+1=1// 0+1+...+2=3// 0+1+...+3=6// 0+1+...+4=10// 0+1+...+5=15// 0+1+...+6=21// 0+1+...+7=28// 0+1+...+8=36// 0+1+...+9=45fmt.Printf("0+1+...+%d=%d\n", i, a(i))}}

sum不是函数体里面定义的,是一个自由变量,自由变量就会连一根线连到sum上面,不断的找连接关系,连成一棵树最终会把所有我们要连接的变量都连到,就个就是函数式编程的闭包

其他语言的闭包

python中的闭包

	def adder():sum = 0def f(value):nonlocal sumsum += valuereturn sumreturn f
  • python原生支持闭包
  • 使用_closure_来查看闭包内容

c++中闭包

	auto adder() {auto sum = 0;return [=] (int value) mutable {sum += value;return sum;};}
  • 过去:stl或者boost带有类似库
  • c++11及以后:支持闭包

java中的闭包

    Function<Integer,Integer> adder(){final Holder<Integer> sum = new Holder<>(0);return (Integer value) -> {sum.value += value;return sum.value;};}
  • 1.8以后:使用Function接口和Lambda表达式来创建函数对象
  • 匿名类或Lambda表达式均支持闭包

go语言闭包应用

斐波那契数列

package mainimport "fmt"func fibonacci() func() int {a,b:=0,1return func()int{a,b = b,a+breturn a}
}
func main() {f := fibonacci()fmt.Println(f()) // 1fmt.Println(f()) // 1fmt.Println(f()) // 2fmt.Println(f()) // 3fmt.Println(f()) // 5fmt.Println(f()) // 8fmt.Println(f()) // 13fmt.Println(f()) // 21
}

为函数实现接口

package mainimport ("bufio""fmt""io""strings"
)func fibonacci() intGen {a, b := 0, 1return func() int {a, b = b, a+breturn a}
}type intGen func() intfunc (g intGen) Read(p []byte) (n int, err error) {next := g()if next > 10000 {return 0, io.EOF}s := fmt.Sprintf("%d\n", next)return strings.NewReader(s).Read(p)
}func printFileContents(reader io.Reader) {scanner := bufio.NewScanner(reader)for scanner.Scan() {fmt.Println(scanner.Text())}
}func fib(n int) int {var res intf := fibonacci()for i := 1; i < n; i++ {res = f()}return res
}
func main() {f := fibonacci()// 1// 1// 2// ...printFileContents(f)
}

使用函数遍历二叉树

package mainimport "fmt"type Node struct {Value       intLeft, Right *Node
}func CreateNode(value int) *Node {return &Node{Value: value}
}func (node Node) Print() {fmt.Print(node.Value, " ")
}func (node *Node) SetValue(value int) {if node == nil {fmt.Println("node is nil")return}node.Value = value
}func (node *Node) Traverse() {node.TraverseFunc(func(n *Node) {n.Print()})fmt.Println()
}
func (node *Node) TraverseFunc(f func(node *Node)) {if node == nil {return}node.Left.TraverseFunc(f)f(node)node.Right.TraverseFunc(f)
}func main() {root := Node{Value: 3}root.Left = &Node{Value: 1}root.Right = &Node{5,nil,nil}root.Right.Left = new(Node)root.Left.Right = CreateNode(2)root.Right.Left.SetValue(4)// 1 2 3 4 5root.Traverse()count := 0root.TraverseFunc(func(node *Node) {count++})// 5fmt.Println(count)
}

go语言对闭包的想法更为自然,不像其他的语言语法看起来怪怪的,不需要修饰如何访问自由变量,没有Lambda表达式,但是有匿名函数

更多推荐

Go学习:函数式编程

本文发布于:2023-07-28 19:27:18,感谢您对本站的认可!
本文链接:https://www.elefans.com/category/jswz/34/1287137.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文标签:函数

发布评论

评论列表 (有 0 条评论)
草根站长

>www.elefans.com

编程频道|电子爱好者 - 技术资讯及电子产品介绍!