golang 函数

2018-10-02  本文已影响13人  one_zheng

 在golang中,函数不仅可以用于封装代码、分割功能、解耦逻辑,还可以化身为普通的值,在其他函数间传递、赋予变量、做类型判断和转换等等,就像切片和字典的值那样。

函数值可以由此成为能够被随意传播的独立逻辑组件(或者说功能模块)。

demo:

package main

import "fmt"

type Printer func(content string) (n int, err error)

func printToStd(content string) (byteNum int, err error) {
        return fmt.Println(content)
}

func main() {
  var p Printer
  p = printToStd
  p("something")
}

  demo中,先声明了一个名叫Printer的函数类型。函数签名(就是函数的参数列表和结果列表)定义了可用来鉴别不同函数的那些特征,同时定义了我们与函数的交互方式。

 注意,各个参数和结果的名称不能算作函数签名的一部分,甚至对于结果声明来说,没有名称都可以。

 只要函数签名是一致的,就可以说它们是一样的函数,或者说实现了同一个函数类型的函数。

Go语言在语言层面支持了函数式编程。

问:什么是高阶函数?
答:高阶函数满足两个条件: 1.接受其他函数作为参数传入; 2.把其他的函数作为结果返回。
  只要满足了其中一点,我们就可以说这个函数是一个高阶函数。高阶函数也是函数式编程中的重要概念和特征。

demo1: 1.接受其他函数作为参数传入

 编写calculate函数来实现两个整数间的加减乘除运算,但是希望两个整数和具体的操作都有该函数调用方给出。

  type operate func (x, y int) int

  func calculate(x int, y int, op operate) {
    if op == nil {
        return 0, errors.New("invalid operation") 
    }
    return op(x,y), nil
  }

  calculate函数的其中一个参数是operate类型的,而且后者就是一个函数类型。在调用calculate函数的时候,我们需要传入一个operate类型的函数值。
  只要传入的函数与operate的签名一致,并且实现得当就可以了

  op := func(x, y int) {
    return x + y
  }
demo2: 2.把其他的函数作为结果返回。
  x,y = 56, 78 
  add := genCalculator(op)
  result, err = add(x, y)
  fmt.Println("The result : %d (error: %v)\n",result, err)

func  genCalculator(op operate ) func(int,int) (int,error) {
   return  func(x int, y int) {
     if op == nil {
        return 0, errors.New("invalid operation") 
     }
      return op(x,y), nil
   }
}
什么是闭包?

  在一个函数中存在对外来标识符的引用(既不代表当前函数的任何参数或结果,也不是函数内部声明的,是直接从外边拿过来的),这中变量叫自由变量闭包就是因为引用了自由变量,而呈现出一种“”不确定“的状态,也叫“”开放”状态。也就是说,它的内部逻辑并不是完整的,有一部分逻辑需要这个自由变量参与完成,而后者代表了什么在闭包函数被定义的时候确实未知的。

  在demo2中,op就是自由变量,当Go语言读取到if op == nil这一行时会试图去寻找op所代表的的东西,它会发现op代表的是genCalculator函数的参数,然后,它会把这两者联系起来,这时可以说,自由变量op被捕获了。

闭包的意义?

  表面上看我们只是延迟了实现一部分程序逻辑或功能而已。但实际上,我们是动态地生成那部分逻辑功能。我们可以借此在程序运行的过程中,根据需要生成功能不同的函数,继而影响后续的程序行为。这与GoF设计模式中的模板方法有着异曲同工之妙。

传入函数的参数值后来怎么样了?

demo3:

package main 
import "fmt"

func main() {
        array1 := [3]string{"a", "b", "c" }
        fmt.Printf("The array: %v\n",array1)
        array2 := modifyArray(array1)
        fmt.Println("The modified array: %v\n", array2)
        fmt.Println("The original array: %v\n", array1)
}

func modifyArray(a [3]string) [3]string {
    a[1] = "x"
    return a
}

答:原数组array1不会改变。所有传给函数的参数值都会被复制,函数在其内部使用的并不是参数值的原值,而是它的副本。注意:对于引用类型:比如:切片、字典、通道,复制的是它们本身么不是它们引用的底层数据,所以当改变时会造成底层数据的改变。

上一篇 下一篇

猜你喜欢

热点阅读