golang 函数

栏目: Go · 发布时间: 7年前

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

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


以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

Sprint

Sprint

Jake Knapp、John Zeratsky、Braden Kowitz / Simon & Schuster / 2016-3-8 / GBP 14.60

媒体推荐 “Every business leader I know worries about the same thing: Are we moving fast enough? The genius of Jake Knapp’s Sprint is its step-by-step breakdown of what it takes to solve big problems an......一起来看看 《Sprint》 这本书的介绍吧!

JSON 在线解析
JSON 在线解析

在线 JSON 格式化工具

XML、JSON 在线转换
XML、JSON 在线转换

在线XML、JSON转换工具

HSV CMYK 转换工具
HSV CMYK 转换工具

HSV CMYK互换工具