当前位置: 首页 > news >正文

【Hello Go】Go语言函数

Go语言函数

    • 定义格式
    • 自定义函数
      • 无参数无返回值
      • 有参数无返回值
      • 不定参数列表
      • 有返回值
      • 有多个返回值
    • 函数类型
    • 匿名函数和闭包
    • 延迟调用defer
      • defer和匿名函数结合使用
    • 获取命令行参数

定义格式

函数是构成代码执行的逻辑结构

在Go语言中 函数的基本组成为

  • func关键字
  • 函数名
  • 参数列表
  • 返回值
  • 函数体
  • 返回语句

基本代码格式如下

func // 函数名(// 参数) (// 返回值) {                                                                                  // 函数语句// 可以return多个返回值
} 

函数定义说明:

  • func关键字 : Go语言声明函数必须要使用func关键字
  • 函数名称 : Go语言的函数名称默认规则为 开头为小写字母即为私有 否则即为公有
  • 参数列表: 支持多个参数 多个参数之间用逗号分隔开 不支持默认参数
  • 返回类型: 我们可以有多个返回类型
  • 返回值: 如果有返回值 我们必须要添加return语句

自定义函数

无参数无返回值

下面是无参数无返回值函数的定义和调用

// 无参数无返回值函数的定义
func test()  {fmt.Println("hello world")
}func main()  {// 调用test()
}

有参数无返回值

// 有参数无返回值函数的定义 
func test2(a , b int) {}func test3(a int , b int)  {}

这里的定义方式有两种 一种是每个标识符后面都添加数据类型 一种是将数据类型添加在最后(如果都一样的话)

func main()  {// 调用test2(10 , 10)
}

不定参数列表

不定参数是指函数传入的参数不确定 为了做到这点 我们首先要将函数定义为接受不定参数类型

定义代码如下

func test4(args ...int)  { for _, v := range args {fmt.Println(v)}
}

形如 ... type 格式类型只能作为函数的参数类型存在 并且只能作为最后一个参数

	// 函数调用 可以传递0~多个数据test4()test4(1)test4(1 , 2, 3, 4)

不定参数的传递

不定参数也会遇到需要继续往下传递参数的情况 下面是不定参数传递的两段代码

func test4(args ...int)  { for _, v := range args {fmt.Println(v)}
}func test5(args ...int)  { for _, v := range args {fmt.Println(v)}
}func test6(args ...int){test4(args ...)      // 传递方式一 直接将所有参数全部传入 格式如图test5(args[1:]...)   // 传递方式二 将参数列表中从1开始(包括1位置)全部传入
}

在我们的test6函数中演示了 用不定参数传参数的两种方式

方式二中我们使用了切片 不了解的同学可以暂时放放 下面几篇博客中会进行讲解

这里我们只需要记住传参的格式是 args ...

有返回值

我们的返回值定义在参数后面

虽然说我们在定义返回值的时候可以省略标识符 直接使用类型 但是官方文档确不推荐我们这么做 因为这样子做会导致我们程序的可读性变差

我们推荐下面这种定义方式

func test7(key int) (value int)  {return 1
}

如果一个函数有返回值 那么我们在函数的最后就必须要返回一个值 否则会编译不通过

有多个返回值

在函数有多个返回值的情况下 我们有两种返回方式

方式一 : 给各个返回值标识符命名 之后return


func test9(key int) (a1 int , a2 int)  {a1 = 1 a2 = 2return 
}

方式二: 直接return多个返回值

func test8(key int) (a1 int , a2 int)  {return 1 , 2
}

函数类型

在Go语言中 函数也是一种数据类型

我们可以通过type来定义它 它的类型就是拥有相同参数 相同返回值的类型

我们可以用它来做到一些好玩的事情 比如说函数套函数

我们下面定义了一个函数类型 给他取别名为 FuncType

type FuncType func(int , int) (int)  // 声明了一个函数类型 注意 func后面没有函数名 

那么我们就可以在下面的函数中 使用这个FuncType作为参数

type FuncType func(int , int) (int)  // 声明了一个函数类型 注意 func后面没有函数名 func Clac(a int , b int , f FuncType)(result int){result = f(a  , b)return result
}func Add(a int , b int) (result int){result = a + b return result
}

之后我们就可以这么调用这个函数

res := Clac(10 , 8 , Add)

匿名函数和闭包

这里先给大家解释下闭包的概念

闭包就是一个函数 捕获 了和他在同一作用域的其他变量和常量

这也就意味着 当一个函数闭包了 不管这个函数在任何地方被调用 它都能使用这些变量和常量 不管它们有没有出作用域

所以说 只要闭包还在使用 这些变量就会一直存在 不会销毁


在Go语言中 所有的匿名函数都是闭包的

下面简单介绍几种匿名函数的定义方式 解释就直接放在注释里面了

	// 这两个参数会被我们下面的匿名函数捕获var a int = 10var str string = "abcde"// 方式一 使用:= 来让一个变量接收func类型f1 := func() { // 这里的func()是一个匿名函数 无参数无返回值fmt.Println(a, "  ", str)}
	// 方式二 在你们函数的末尾直接调用func(a int , b int)(result int){result = a + bfmt.Println(result)return result}(1 , 1)

匿名函数的闭包是引用

如果我们在匿名函数内部 修改了闭包的变量 那么外部的值也会改变

	var a int = 10 var str string = "abcde"f1 := func(){a = 20str = "go"}f1()// 此时打印a 和 str 我们会发现它们的值改变了 fmt.Println(a , str)	
}

返回值为匿名函数

我们都知道 局部变量在出了作用域之后就会被销毁 但是我们如果使用匿名对象作为返回值 就能够让该变量存在的时间延长 比如说下面的代码

func squares() (func() int) {var x int return func() (ans int){x++return x * x}
}func main()  {f := squares()fmt.Println(f())fmt.Println(f())fmt.Println(f())fmt.Println(f())
}

解释下 我们 squares 函数的返回值是一个匿名对象

该匿名对象将局部变量 x 捕捉了 并且在内部++ 之后平方

因此 该局部变量的生命周期被延长了 所以说我们最后得到的结果会是

1 4 9 16 … …

延迟调用defer

关键字defer用于延时一个函数或者是方法的调用

需要注意的是 defer方法只能出现在函数的内部

func test() {defer fmt.Println("no.1")fmt.Println("no.2")
}func main() {test()defer fmt.Println("no.3")fmt.Println("no.4")// 输出结果为  2  1  4  3
}

上面的两个函数演示了defer关键字的作用

在一个作用域内 如果我们使用了defer关键字 那么语句就会在作用域即将被销毁的时候执行

如果说有多个defer语句的话 遵循栈的原则 也就是后进先出

这里需要注意的一点是

如果有某个函数或者某个延时调用发生错误 整个栈也会被清空 也就是所有延时调用语句都会执行

func test() {defer fmt.Println("no.1")fmt.Println("no.2")
}func test1(x int) {v1 := 100 / x_ = v1 
}func main() {test()defer fmt.Println("no.3")fmt.Println("no.4")defer test1(0) // 会发生错误// 输出结果为  2  1  4  3
}

就比如说上面的代码 我们故意写了除0错误 但是它的执行结果却是所有语句都运行完毕之后再报错

defer和匿名函数结合使用

func main() {a , b := 10 , 20 defer func(x int) {fmt.Println(x)} (a) // 将a以传值传递的方式传递给匿名函数funca += 10 b += 100fmt.Println(a)fmt.Println(b)// 输出为 20  120  10
}

获取命令行参数

在Go语言中 如果我们要获取命令行参数的话 需要使用到os包

代码演示如下

func main()  {args := os.Args // 获取用户的所有参数// 如果获取失败 或者是参数不足则报错 否则打印出前两个参数if args == nil || len(args) < 2{fmt.Println("err!!")return}ip := args[1]port := args[2]fmt.Println(ip)fmt.Println(port)
}
http://www.lryc.cn/news/231422.html

相关文章:

  • docker小技能:容器IP和宿主机IP一致( Nacos服务注册ip为内网ip,导致Fegin无法根据服务名访问 )
  • Android笔记:震动实现
  • CSDN每日一题学习训练——Java版(二叉搜索树迭代器、二叉树中的最大路径和、按要求补齐数组)
  • WPF中有哪些布局方式和对齐方法
  • 【2012年数据结构真题】
  • k8s_base
  • 2023年亚太杯APMCM数学建模大赛数据分析题MySQL的使用
  • 自学SLAM(8)《第四讲:相机模型与非线性优化》作业
  • STL—next_permutation函数
  • Mysql 三种不使用索引的情况
  • Ladybug 全景相机, 360°球形成像,带来全方位的视觉体验
  • centos 6.10 安装swig 4.0.2
  • mask: rle, polygon
  • 【JMeter】JMeter压测过程中遇到Non HTTP response code错误解决方案
  • 【Kingbase FlySync】评估工具安装及使用
  • pandas教程:Data Aggregation 数据聚合
  • 开启创造力之门:掌握Vue中Slot插槽的使用技巧与灵感
  • 【算法练习Day48】回文子串最长回文子序列
  • ubuntu下C++调用matplotlibcpp进行画图(超详细)
  • 芯科科技推出新的8位MCU系列产品,扩展其强大的MCU平台
  • Flink CDC
  • 数据结构-链表的简单操作代码实现3-LinkedList【Java版】
  • JTS: 24 MinimumDiameter 最小矩形
  • MacOS Ventura 13 优化配置(ARM架构新手向导)
  • 多区域OSPF配置
  • 【强化学习】day1 强化学习基础、马尔可夫决策过程、表格型方法
  • openwrt Docker不能联网
  • EtherCAT从站EEPROM组成信息详解(2):字8-15产品标识区
  • SpringBoot--中间件技术-4:整合Shiro,Shiro基于会话SessionManager实现分布式认证,附案例含源代码!
  • 【QT基础入门】QT中的容器类