首页 > 其他分享 >Go语言函数详解

Go语言函数详解

时间:2023-10-27 11:14:03浏览次数:33  
标签:函数 int fmt Println 详解 func Go main

函数

(1)函数的定义

  • 函数使用func进行定义
  • 函数是基本的代码块,用于执行一个任务
  • Go语言至少有一个main函数
  • 函数声明告诉了编译器函数的名称,返回类型和参数
//1.无参数无返回值函数的定义
func test1(){
    fmt.Println("无参数无返回值函数的定义\n")
}

//2.有参数无返回值函数的定义
func test2(str string){
    fmt.Println("有参无返回值:%s\n",str)
}

//3.多参无返回值
func test3(a,b int) int{
    return a+b
}

//4.多个返回值
func test4(a,b int) (int,int){
    return a,b
}

//函数的调用
fun main()
{
    test1()
    test2("WY")
    test3(1,2)
    test4(5,6)
    
}

(2)函数的可变参数

  • 一个函数的参数类型确定,但个数不确定时可以使用可变参数
  • 定义可变参数时,使用**... **符号
  • 可变参数本质上是保存到了一个数组当中
  • 注意:可变参数要放在参数定义的最后;一个函数的参数列表最多只有一个可变参数
//可变参数函数定义
func add_sum(num...int){
    sum := 0
    for _,i := range num{
        sum += num[i-1]
    }
}

(3)参数传递的类型

  • 参数分为两种类型:

    • 值类型-操作的是数据本身,例如我们常用的:int,string,bool,float64,array
    • 引用类型的数据-操作的是数据的地址:例如slice,map,chan
  • 值传递在调用完函数后不会修改原本值传递参数的值

//定义一个函数,进行值传递实验
func update(arr2 [4]int){
    fmt.Println("arr2接收到的数据:"arr2)        
    arr2[0] = 100
    fmt.Println("arr2修改后的数据:"arr2)
}

//定义主函数,调用update进行值传递观察
func main(){
    arr := [4]int{1,2,3,4}    //定义一个数组
    fmt.Println(arr)
    //传递值
    update(arr)
    fmt.Println("调用后的数据")     //这里的结果为[1,2,3,4]
}

  • 引用类型数据的参数传递在调用完函数后会修改原本值传递参数的值
package main

import "fmt"

/*
=====引用传递=======
1、定义一个切片
2、通过方法修改切片里面的数据
3、打印输出修改后值
*/
func main() {
	arr := []int{1, 2, 3, 4}       //定义一个切片
	fmt.Println("调用修改前的数据", arr)
	updata2(arr)
	fmt.Println("调用修改后的数据", arr)
}
func updata2(arr []int) {
	fmt.Println("arr接受的数据:", arr)
	arr[0] = 100
	fmt.Println("arr修改后的数据:", arr)
}
/*
调用修改前的数据 [1 2 3 4]
arr接受的数据: [1 2 3 4]
arr修改后的数据: [100 2 3 4]
调用修改后的数据 [100 2 3 4]
*/

(4)函数变量的作用域

  • 作用域:变量可以使用的范围
  • 局部变量:函数内部定义的变量,叫做局部变量,只能在定义该变量的函数内使用
  • 全局变量:函数外部定义的变量,叫做全局变量,可以在任何函数里面使用
package main

import "fmt"

// 全局变量  所有函数都可以使用
var num int = 30

func main() {
   // 局部变量 temp 只能在定义temp的方法中使用
   temp := 100

   if b := 1; b < 10 {
      temp := 30
      fmt.Println("局部变量b:", b)
      fmt.Println("局部变量temp:", temp)
   }
   fmt.Println(temp)

   // 调用f1,f2方法
   f1()
   f2()
}
func f1() {
   fmt.Println(num)
}
func f2() {
   fmt.Println(num)
}

(5)递归函数

  • 递归函数的定义:可以自己调用自己的函数
  • 递归函数必须要有一个出口,否则会陷入死循环
package main

import "fmt"

/*
1、创建一个求和函数 getsum
2、给递归函数一个出口
3、创建对象来接收函数
*/
func main() {
	//3、创建对象来接收函数
	sum := getSum(5)

	fmt.Println(sum)
}

//1、创建一个求和函数 getsum
func getSum(n int) int {

	//2、给递归函数一个出口
	if n == 1 {
		return 1
	}
	return getSum(n-1) + n
}

(6) defer延迟函数

  • defer函数可以延迟一个函数的执行

在以下代码块中,控制台打印的结果是1 2 4 3,因为defer将f(3)延迟到最后执行了

package main

import "fmt"

func main() {

   f("1")
   f("2")
   defer f("3")           //将该函数延迟到最后执行
   f("4")
}
func f(s string) {
   fmt.Println(s)
}

  • 如果使用了多个defer语句,则函数执行到最后时,这些defer语句会按照逆序执行
package main

import "fmt"

func main() {

   f("1")
   f("2")
   defer f("3")
   f("4")
   defer f("5")
   f("6")
   defer f("7")
   f("8")
}
func f(s string) {
   fmt.Println(s)
}
/*
输出结果:
1
2
4
6
8
7
5
3
*/

(7)func数据类型详解

  • 在go语言中,函数是复合类型,本质上可以看作是一个特殊的变量

  • func本身就是一个数据类型,func定义的函数之间可以相互赋值

package main

import "fmt"

/*
fun本身就是一个数据类型
1、创建一个函数
2、在mian方法定义相同类型的函数
3、将定义后的函数赋值个另一个函数
*/
func main() {
	//2、在mian方法定义相同类型的函数
	var fg func(int, int)

	//3、将定义后的函数赋值个另一个函数
	fg = ff              //将fg函数定义为ff函数相同的功能
	fg(1, 2)             //最后输出的结果为 1   2

}

//1、创建一个函数
func ff(a, b int) {
	fmt.Println(a, b)
}



(8)匿名函数

  • 什么是匿名函数:匿名函数就是没有名字的函数

匿名函数的创建

//这就是一个匿名函数,但如果直接放在代码块中不调用则编译器会报错
func() {
    fmt.Println("匿名函数")
}

//带参匿名函数
func(a,b int){
    fmt.Println("带参匿名函数")    
}

//带参且具有返回值函数
func(a,b int) int{
    fmt.Println("带参和返回值匿名函数")
    return a+b
}

匿名函数多种调用方式

func main(){
    
    //方式1:用一个变量调用匿名函数
    f3 := func(){
        fmt.Println("匿名函数的调用方式1")
    }
    f3() //调用匿名函数
    
    //方式2:创建匿名函数后直接调用func(),在{}后直接添加()进行调用
    func(){
        fmt.Println("匿名函数调用方式2")
    }()
    
    //带参匿名函数的调用
    func(a,b int){
        fmt.Println("带参匿名函数调用")
    }(1,2)    
    
    //带参和返回值匿名函数调用
    f5 := func(a,b int) int{
        fmt.Println("带参和返回值匿名函数")
        return a+b
    }(1,2)    
    fmt.Println(f5)     //打印调用结果
    
}

(9)高阶函数和回调函数

  • 回调函数:将一个函数作为另一个函数的参数

  • 若将fun1函数作为fun2函数的参数,则fun2叫做高阶函数,fun1称为回调函数

package main

import "fmt"

//1、创建一个高阶函数oper,传如一个函数类型的参数且有返回值
func oper(a, b int, fun func(int, int) int) int {
	r2 := fun(a, b)
	return r2
}

//2.创建其他方法的函数
func add(a, b int) int {
	return a + b
}

//创建其他方法的函数
func sub(a, b int) int {
	return a - b
}

func main() {
	r1 := add(1, 2)
	fmt.Println(r1)

	//3、调用高阶函数时,传入其他方法的函数
	r3 := oper(3, 4, add)
	fmt.Println(r3)
	r4 := oper(8, 4, sub)
	fmt.Println(r4)
}





(10)闭包

  • 一个外层函数中,有内层函数,该内层函数中,会操作外层函数的局部变量并且该外层函数的返回值就是这个内层函数
  • 这个内层函数和外层函数的局部变量,统称为闭包结构。
  • 局部变量的生命周期就会发生改变,正常的局部变量会随着函数的调用而创建,随着函数的结束而销毁但是闭包结构中的外层函数的局部变量并不会随着外层函数的结束而销毁,因为内层函数还在继续使用
package main

import "fmt"

// 定义一个闭包函数。func() int是局部变量和返回值fun的类型
func increment() func() int {

	//局部变量
	i := 0
	//定义一个匿名函数 给变量自增并返回
	fun := func() int {
		//内层函数,没有执行
		i++
		return i
	}
	return fun
}

func main() {
	//定义一个变量 接受返回值(函数)
	r1 := increment()      //创建函数
	fmt.Println(r1)       //这里打印的是地址,因为r1还没有执行

	v1 := r1()
	fmt.Println(v1)
	//不断调用自增+1
	v2 := r1()
	fmt.Println(v2)
	//不断调用自增+1
	fmt.Println(r1())
	fmt.Println(r1())
	fmt.Println(r1())
	fmt.Println(r1())
	fmt.Println("====================")
	//	==========================================================
	r2 := increment()      //创建函数,i也会清零
	v3 := r2()
	fmt.Println(v3)
	fmt.Println(r2())
	fmt.Println(r2())
	fmt.Println(r2())
	fmt.Println(r1())     //这里调用的是r1,因此i值还是原来r1的i值,这里自增后为7
	fmt.Println(r2())
}


/*
输出结果:
0xd5ea00
1
2
3
4
5
6
====================
1
2
3
4
7
5

*/

标签:函数,int,fmt,Println,详解,func,Go,main
From: https://www.cnblogs.com/sillyfox/p/17791762.html

相关文章

  • 【技术文档】RTSP协议详解
    1 概要RTSP(RealTimeStreamingProtocol)实时流协议:一种流媒体控制协议,可对流媒体进行暂停、快进、快倒等操作。流媒体就是实时在线点播。而流媒体与普通媒体的差别在于:对于普通媒体,在访问它之前要得到全部的内容;对于流媒体,则在完全接收到全部内容之前就开始访问。本文主要介......
  • ogg 列映射函数使用方法汇总
    MAPsource.Table_name,targettarget.Table_name,exceptionsonly,INSERTALLRECORDS,KEYCOLS(Column1),colmap(usedefaults,gg_dml_type=@GETENV("LASTERR","OPTYPE"),gg_db_error=@GETENV("LASTERR","DBERRNUM"));......
  • MongoDB 中的锁分析
    MongoDB中的锁前言MongoDB中锁的类型锁的让渡释放常见操作使用的锁类型如果定位MongoDB中锁操作1、查询运行超过20S的请求2、批量删除请求大于20s的请求3、kill掉特定client端ip的请求4、查询所有wait锁定的写操作5.返回索引的创建信息总结参考......
  • Django实战项目-学习任务系统-任务管理
    接着上期代码框架,开发第3个功能,任务管理,再增加一个学习任务表,用来记录发布的学习任务的标题和内容,预计完成天数,奖励积分和任务状态等信息。 第一步:编写第二个功能-用户注册1,编辑模型文件:./mysite/study_system/models.py:classStudyTask(models.Model):task_id=mod......
  • 常用函数一览
    一、abs()<cstdlib>(或<stdlib.h>)中的abs函数原型:intabs(intx)返回类型int,参数类型int。用于计算整数的绝对值在<cmath>中的abs函数abs函数是一个重载函数,可以处理多种数值类型的绝对值原型doubleabs(doublenum);floatabs(floatnum);longdoubleabs(longd......
  • 云端golang开发,无需本地配置,能上网就能开发和运行
    欢迎访问我的GitHub这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos需求学习golang的时候,需要一个IDE,还需要一个能运行程序的环境,以及一个MySQL数据库对于有经验的程序员来说,自己动手安装部署即可,但是小白和懒人也是存在的...背景背......
  • 给google浏览器配独立ip
     参数为:--proxy-server="127.0.0.1:1234"google浏览器中的属性: 类似如下:C:\ProgramFiles\Google\Chrome\Application\chrome.exe"--user-data-dir=D:\chrome\102--proxy-server="127.0.0.1:1234"......
  • 基于Googlenet深度学习网络的信号调制类型识别matlab仿真
    1.算法运行效果图预览   2.算法运行软件版本MATLAB2022a 3.算法理论概述      信号调制类型识别是在无线通信和无线电频谱监测中的一个重要任务。不同信号调制类型具有不同的频谱特征,深度学习方法在信号调制类型识别中取得了显著的成果。 3.1深度学习与......
  • mongoose 的使用
    constmongoose=require('mongoose');mongoose.connect('mongodb://admin:[email protected]:27017/GIS?authSource=admin',{useUnifiedTopology:true,useNewUrlParser:true,}).then(......
  • Google三驾马车之二:MapReduce
    第一次接触mr还是在入门mit6.824的lab1,最近重新读了一遍原始论文,又有了一些新的想法,简单做一些记录。作为Google分布式系统的重要组成,本篇文章核心在于map/reduce操作带来的抽象并行化,给出接口之后,编写应用程序的程序员就不需要对底层的机制做过多的处理。而在本质上,mr只是实现了......