0


【30天熟悉Go语言】9 Go函数全方位解析

作者:秃秃爱健身,多平台博客专家,某大厂后端开发,个人IP起于源码分析文章 😋。
源码系列专栏:Spring MVC源码系列、Spring Boot源码系列、SpringCloud源码系列(含:Ribbon、Feign)、Nacos源码系列、RocketMQ源码系列、Spring Cloud Gateway使用到源码分析系列、分布式事务Seata使用到源码分析系列、JUC源码系列
基础系列专栏:30天熟悉GO语言(建设中)
码文不易,如果感觉博主的文章还不错,请点赞👍、收藏 ⭐️支持一下博主哇 🙏
联系方式:Saint9768,加我进技术交流群,一起学习进步📚、早日开启养老模式✈️🌊

文章目录

一、前言

在这里插入图片描述

Go系列文章:

  1. GO开篇:手握Java走进Golang的世界
  2. 2 Go开发环境搭建、Hello World程序运行
  3. 3 Go编程规约和API包
  4. 4 Go的变量、常量、运算符
  5. 5 Go 基本数据类型
  6. 6 Go 复杂数据类型之指针
  7. 7 Go流程控制之分支结构if、switch
  8. 8 Go流程控制之循环结构for range、goto、break、continue

Go专栏传送链接:https://blog.csdn.net/saintmm/category_12326997.html

二、函数

1、函数简介

1)函数是什么?

函数是一个基本的代码块,是用于完成某一功能的程序指令集;可以对特定的功能进行提取,形成一个代码片段,这个代码片段即函数。

我们可以通过函数划分不同的功能,逻辑上每个函数执行的是指定的任务。

2)为什么要使用函数?

提高代码的复用性,减少代码的冗余,提高代码的可维护性。

3)函数的特点?

  • 支持可变参数
  • 支持多返回值
  • 支持匿名函数
  • 函数也是一种数据类型,可以赋值给变量
  • 不支持函数的重载(overload),即:一个包不能有两个名字一样的函数
  • 不支持嵌套(nested),即:一个包不能有两个名字一样的函数

2、函数的声明和使用

1)函数声明

函数的定义格式如下:

funcname([parameter_list])[return_types]{
   函数体
}
  • 函数func关键字声明,包含一个函数名、参数列表、返回值列表和函数体。 - 左大括号要和func关键字在一行,不能另起一行。
  • name:函数名称,只能定义一次,函数名称是唯一的,不支持重载- 函数名命名规范:驼峰命名- 首字母不能是数字;- 首字母大写该函数可以被本包文件和其它包文件使用(类似Java方法的public权限) - 首学母小写只能被本包文件使用,其它包文件不能使用(类似Java方法的private权限)
  • parameter_list:参数列表,指定参数类型、参数之间的顺序、参数个数; - 参数类型在参数名之后,函数可以没有参数、多个参数。- 当多个连续的参数是同一类型,除了最后一个参数写参数类型之外,可以参数都可以省略。
  • return_types:返回类型列表,函数可以返回任意数量的返回值; - 返回多个值时,多返回值必须用括号括起来,并以逗号隔开。- 返回单个值时,直接相应的返回值数据类型。- 如果函数没有返回值,返回类型列表可以省略。
  • 函数体:代码集合。

示例1:两数相加

funcaddAndFormat(num1, num2 int)int{
    sum := num1 + num2
    return sum
}

示例2:两数相加,返回相加结果和结果说明

funcaddAndFormat(num1, num2 int, format string)(int,string){
    sum := num1 + num2
    return sum, fmt.Sprintf(format, sum)}

函数名不支持重载

在这里插入图片描述

1> 返回多个值 – 返回值无名称

格式:

funcname(arg1 T, arg2 T)(T, T){...return r1, r2
}

特点:

  • 当需要返回两个值及以上时,返回类型用小括号包裹,逗号分隔。
  • return语句中携带多个返回值。

2> 返回多个值 – 返回值有名称

格式:

funcname(arg1 T, arg2 T)(r1 T, r2 T){...return}

特点:

  • 返回值类型指定名称后,return语句中,可以不带值,也可以都带上。
  • 不过,当返回值有了名称之后,即使是只有一个返回类型,也需要用小括号包裹。

return可以不用携带值的原因?

  • 因为等价于在返回时,初始化好了返回值;
  • 以上面的格式来看,r1r2 是两个初始化的变量; - 在函数运算中,只要将返回结果存入r1r2中;- 如果不存,也可以选择在return语句携带多个返回值。

2)函数调用

函数可以被多次调用,在函数调用时传递的参数为实际参数(

实参

) ,其有具体的值,用来给函数形式参数(

形参

) 传递数据。

格式:

r1, r2 :=name(param1, param2, param3)
  • 格式中,r1, r2表示两个返回值;param1, param2, param3表示三个参数。

如果接收多个值时,某个值不想使用,可以选择使用关键字

_

替代,表示不使用这个返回值。

r1,_:=name(param1, param2, param3)

以函数声明中的示例2为例:

1> 返回值全部接收:

package main

import"fmt"funcmain(){
    format :="two num add, the result is : %d "
    sum, str :=addAndFormat(1,2, format)
    fmt.Println(sum)
    fmt.Println(str)}// 两数相加,返回相加结果和结果说明**funcaddAndFormat(num1, num2 int, format string)(int,string){
    sum := num1 + num2
    return sum, fmt.Sprintf(format, sum)}

控制台输出:
在这里插入图片描述

2> 只接收一个返回值:

package main

import"fmt"funcmain(){
    format :="two num add, the result is : %d "
    sum,_:=addAndFormat(1,2, format)
    fmt.Println(sum)}

3、函数参数

1)值传递

Go语言中,基本数据类型和数组默认都是值传递的,即:进行值拷贝;在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改时,将不会影响到原来的值。

示例:定义一个swap函数,用于交换两个int类型数据的值

package main

import"fmt"funcmain(){
    a :=3
    b :=6
    fmt.Printf("交换前 a 的值为 : %d\n", a)
    fmt.Printf("交换前 b 的值为 : %d\n", b)swap(a, b)
    fmt.Printf("交换后 a 的值为 : %d\n", a)
    fmt.Printf("交换后 b 的值为 : %d\n", b)}// 两值交换funcswap(x, y int)int{var temp int

    temp = x
    x = y
    y = temp

    return temp
}

控制台输出:
在这里插入图片描述

调用swap()函数交换a,b值,交换前后 a,b的值没有改变;

所以值传递不会改变传入函数实参的值,它只是复制一份数据用于函数体的执行。

2)引用传递

如果希望在函数内的变量能修改函数外的变量,可以传入变量的地址

&

,函数内以指针的方式操作变量;从效果来看类似

引用传递

示例:

package main

import"fmt"funcmain(){
    a :=3
    b :=6
    fmt.Printf("交换前 a 的值为 : %d\n", a)
    fmt.Printf("交换前 b 的值为 : %d\n", b)swap(&a,&b)
    fmt.Printf("交换后 a 的值为 : %d\n", a)
    fmt.Printf("交换后 b 的值为 : %d\n", b)}// 两值交换funcswap(x, y *int)int{var temp int

    temp =*x
    // 将y地址上的值复制到x的地址上*x =*y
    *y = temp

    return temp
}

控制台输出:
在这里插入图片描述

PS:

  • 无论是值传递,还是引用传递,传递给函数的都是变量的副本;不过,值传递的是值的拷贝,引用传递的是地址的拷贝。 - 一般而言,地址拷贝更为高效。值拷贝的效率取决于拷贝对象的大小,对象越大,性能越低。
  • map、slice、chan、指针、interface默认以引用的方式传递。

3)不定参数 / 可变参数

不定参数传值指:函数的参数类型固定、参数数量不固定;Go语言中可变参数本质上是

slice

,并且该 slice 只能有一个,且必须是最后一个。

// 0个或多个参数funcmyfunc1(args ...int){}// 1个或多个参数funcadd(a int, args…int)int{}// 2个或多个参数funcadd(a int, b int, args…int)int{}
  • 注意:其中 args 是一个slice,我们可以通过 arg[index] 依次访问所有参数,通过 len(arg) 来判断传递参数的个数;

4、函数嵌套/递归使用

1)函数嵌套使用

函数中调用其他函数。
示例:test函数中调用test1函数

package main

import"fmt"funcmain(){test(3,6)}functest1(a, b int){
    fmt.Println(a + b)}functest(a int, b int){
    fmt.Println("into function test")test1(a, b)}

不定参不能直接传递给另外一个不定参

在这里插入图片描述

**解决措施:指定不定参

arr

需要传递到函数

testIndefiniteParam2

的数据个数;**

1> 指定个数:

package main

import"fmt"funcmain(){testIndefiniteParam(1,2,3,4)}// 不定参不能直接传递给另外一个不定参functestIndefiniteParam(arr ...int){// 传递指定个数的数据,不报错// 下标起始0,结束4(不包含4);仅仅为索引4之前的数据testIndefiniteParam2(arr[0:4]...)}functestIndefiniteParam2(arr ...int){
    fmt.Println(arr)}

输出:
在这里插入图片描述
2> 若想要传递全量数据,可以用下列方式:

testIndefiniteParam2(arr[0:]...)

2)递归函数

递归指函数在运行的过程中自己调用自己。

构成递归的条件:

  1. 子问题需要与原始问题是同样的问题,且更加简单;
  2. 递归需要出口,不能无限制地调用本身;在使用递归时,开发者需要设置 退出条件 ,否则递归将陷入无限循环。

示例:斐波那契数列(fibonacci)

package main

import"fmt"funcfibonacci(i int)int{if i ==0{return0}if i ==1{return1}returnfibonacci(i-1)+fibonacci(i-2)}funcmain(){var i intfor i =0; i <10; i++{
        fmt.Printf("%d\n",fibonacci(i))}}

控制台输出:
在这里插入图片描述

5、匿名函数

匿名函数指不需要定义函数名的一种函数实现方式,匿名函数由一个不带函数名的函数声明和函数体组成;在Go语言中,函数可以像普通变量一样被传递 & 使用,并且支持随时在代码里定义匿名函数。

构造函数时,函数没有名称;想调用函数时,需要把匿名函数赋值给一个变量,或 在构造时直接调用。

1)赋值给变量

func1 :=func(arg1 T, arg2 T) T {...return r1
}

这里,

func1

是一个函数类型的变量,可以直接通过

func1(param1, param2)

调用函数。

示例:

package main

import"fmt"funcmain(){// 1> 匿名函数赋值给变量
    func1 :=func(a int, b int)int{
        res := a + b
        fmt.Printf("a + b = %d \n", res)return res
    }func1(2,3)}

2)构造时直接调用

func(arg1 T, arg2 T) T {...return r1
}(param1, param2)

构造函数时,函数声明的右大括号

}

后紧跟要传递的参数(param1, param2);这样,构造完函数后会马上调用。

示例:

package main

import"fmt"funcmain(){//2> 匿名函数构造时直接调用func(a int, b int)int{
        res := a + b
        fmt.Printf("a + b = %d \n", res)return res
    }(2,3)}

6、函数类型

在Go语言中,函数也是一种数据类型,可以将函数赋值给一个变量,则该变量就是一个函数类型的变量了,通过该变量可以对函数调用。(匿名函数中有介绍)

所谓的声明函数类型,实际就是为已存在的函数起一个别名。尤其当我们引用二方包、三方包时,可以给别人的函数取个别名。

语法:

type 自定义数据类型名 func([parameter_list])[return_types]

示例:

package main

import"fmt"// 定义函数类型 为已存在的数据类型起别名type funcTwoParamOneReturn func(int,int)intfuncmain(){// 自定义函数类型使用var f funcTwoParamOneReturn
    f = func2
    fmt.Printf("%T", f)}// func(int, int)funcfunc1(a int, b int){
    fmt.Println(a + b)}// func(int, int) intfuncfunc2(x int, y int)int{
    fmt.Println(x + y)return x + y
}

对已有函数设置别名时,函数的参数类型/个数、返回值类型/个数 需要一一对应,否则会报错;
在这里插入图片描述

7、Go内置函数

和Java语言一样,在Go语言中,有一些函数无需导包即可使用,这样的内置函数有15个:

  1. make:为切片,map、通道类型分配内存并初始化对象。
  2. len:计算数组、切片、map、通道的长度。
  3. cap:计算数组、切片、通道的容量。
  4. delete:删除 map 中对应的键值对。
  5. append:将数据添加到切片的末尾。
  6. copy:将原切片的数据复制到新切片中。
  7. new:除切片、map、通道类型以外的类型分配内存并初始化对象,返回的类型为指针。
  8. complex:生成一个复数。
  9. real:获取复数的实部。
  10. imag:获取复数的虚部
  11. print:将信息打印到标准输出,没有换行。
  12. println:将信息打印到标准输出并换行。
  13. close:关闭通道。
  14. panic:触发程序异常。
  15. recover:捕捉 panic 的异常信息,必须卸载defer相关的代码块中。 - defer将在后面的文章中介绍

三、总结

本文介绍了函数的一些基本概念,比如:函数是什么?为什么要使用函数?函数的特点?怎么声明一个函数?如何调用一个函数?嵌套函数是什么?匿名函数怎么声明使用?Go中内置函数有哪些?

对比Java来看

和Java语言一样:

  • 基础类型都是按值传递,复杂类型都是按引用传递; *Go中基础类型如需按引用传递,需要使用指针。
  • 都可以丢弃返回值。Go中使用关键字_接收返回值用于丢弃,Java直接不用变量接方法的返回。

和Java语言不同的是:

  • Go中返回值可以有多个,Java中多个值需要封装到实体或map返回;
  • Go中的函数不支持重载,而Java方法可以重载;

Java中的方法定义:

访问修饰符 返回值类型 方法名(参数1类型  参数1,参数2类型 参数2...){return 返回值;}

Go中的函数定义:

func 函数名(变量1 变量类型,变量2 变量2类型...)(返回值1 类型1,返回值2 类型2...){return;}
标签: golang

本文转载自: https://blog.csdn.net/Saintmm/article/details/131276748
版权归原作者 秃秃爱健身 所有, 如有侵权,请联系我们删除。

“【30天熟悉Go语言】9 Go函数全方位解析”的评论:

还没有评论