Go语言学习Day4:函数(上)

news2025/1/16 1:03:29

名人说:莫愁千里路,自有到来风。 ——钱珝
创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊)

目录

      • 1、函数的概念与定义
        • ①函数的概念
        • ②函数的具体定义
        • ③多返回值
      • 2、函数参数与作用域
        • ①可变参数
        • ②形式参数与实际参数
        • ③参数的传递细节
        • ④函数作用域
        • ⑤递归函数
      • 3、小结

1、函数的概念与定义

①函数的概念

在Go语言中,函数是基本的代码块,用于执行一个任务。函数可以接受输入参数,并且可以返回一个或多个值。它们是组织和复用代码的基本单位。在Go中,函数也可以被当作变量,传递给其他函数,或者从其他函数返回。

②函数的具体定义

函数的定义包括函数名、参数列表、返回值列表和函数体。格式如下:

//func 函数名 (参数1 类型1,参数2 类型2...)(返回值类型)
func functionName(param1 type1, param2 type2) (returnType) {
    // 函数体
}

案例1:自定义sub函数,并通过样例测试

//创作者:Code_流苏(CSDN)
package main

import "fmt"

func sub(x int, y int) int {
    return x - y
}

func main() {
    result := sub(6, 5)
    fmt.Println("6 - 5 =", result)
}

image-20240326223222601

案例2:加法,函数调用探析

package main

import "fmt"

/*
函数是基本的代码块,用于执行一个任务
*/
//main()主函数 程序的入口
func main() {
    var x int
    var y int
    var z int
    fmt.Println("请输入x和y的值:")
    fmt.Scanf("%d%d", &x, &y)
    //调用函数 函数名()
    z = add(x, y)
    fmt.Println(z)
}

//函数格式
//func 函数名(参数1,参数2...,参数类型) 返回值类型 {
//  函数体
//  return 返回值
//}

func add(a, b int) int {
    c := a + b
    return c
}

image-20240326223332933

③多返回值

Go语言支持函数返回多个值,这在处理错误或者需要返回多种数据时非常有用。

案例1:交换打印

package main

import "fmt"

func swap(x, y string) (string, string) {
    return y, x
}

func main() {
    a, b := swap("hello", "world")
    fmt.Println(a, b)
}

image-20240326223429825

案例2:函数调用,多返回值

package main

import "fmt"

func main() {
	//函数的调用
	printinfo()

	myprint("haha")

	c := add2(2, 3)
	myprintnum(c)

	x, y := swap("yueliusu", "you do it!")
	fmt.Println(x, y)
}

// 无参无返回值的函数
func printinfo() {
	fmt.Println("printinfo")
}

// 有一个参数的函数
func myprint(msg string) {
	fmt.Println(msg)
}

func myprintnum(x int) {
	println(x)
}

/* 有两个参数的函数 */
// 有一个返回值的函数
func add2(a, b int) int {
	c := a + b
	return c
}

// 有多个返回值的函数
func swap(x, y string) (string, string) {
	return y, x
}

image-20240326223622407

2、函数参数与作用域

①可变参数

在Go语言中,函数的参数数量可以是可变的,称为可变参数。通过在参数类型前加上...符号来表示。

案例1:求和

package main

import "fmt"

func sum(nums ...int) {
    total := 0
    for _, num := range nums {
        total += num
    }
    fmt.Println(total) 
}

func main() {
    sum(1, 2)
    sum(1, 2, 3)
}

image-20240326224548809

案例2:不同参数

package main

import "fmt"

func main() {
	getSum("hahah", 1, 2, 3, 4, 5, 6, 7)
}

//在Go中,可变参数通过在参数类型前加上省略号 ... 来指定。
//这种参数在函数内部表现为同类型的切片(slice)。

// 参数 ...参数类型 <------> 可变参数
func getSum(msg string, nums ...int) {
	fmt.Println(msg)
	sum := 0

	for i := 0; i < len(nums); i++ {
		fmt.Println(nums[i])
		sum += nums[i]
	}

	fmt.Println("sum:", sum)
}

//若一个函数的参数有可变参数的同时还有其它参数,可变参数要放参数列表最后的位置
//一个函数的参数列表最多只能有一个可变参数
②形式参数与实际参数
  • 形式参数

    定义函数时,用于接收外部传入值的变量称为形式参数。(接收到)

  • 实际参数

    在调用函数时,传递给函数的实际值或变量称为实际参数。(传递给)

package main

func main() {
    // 形参与实参要一一对应,顺序,个数,类型
    // 实际参数
    println(max(1, 2))
}

//形式参数:定义函数时,用于接收外部传入值的 变量 称为形式参数
//实际参数:在调用函数时,传递给函数的实际值或变量称为实际参数

// max函数,实现两个数值比较大小
// 形式参数
func max(num1, num2 int) int {
    var result int
    if num1 > num2 {
       result = num1
    } else {
       result = num2
    }

    //函数定义时有说明返回值的类型,那么函数中必须使用return语句来返回值
    return result
}

image-20240326224206897

③参数的传递细节

Go语言中函数的参数是通过值传递的,这意味着函数接受的是参数值的一个副本。但是,如果参数是指针、切片或映射等引用类型,则函数可以修改原始数据。

  • 引用传递 ( 切片 )

    引用传递 操作的是数据的地址 如:slice、map、chan…

  • 值传递 ( 数组 )

    值传递 操作的是数据本身 如:基础数据类型int、string、bool、floa64、array、struct…

案例1:引用传递,数据修改

package main

import "fmt"

func modify(s []int) {
    s[0] = 100
}

func main() {
    a := []int{1, 2, 3}
    modify(a)
    fmt.Println(a) // 输出 [100, 2, 3],原始数据被修改
}

image-20240326224746713

案例2:值传递,接收与修改数据

package main

import "fmt"

func main() {
    //值传递

    arr := [4]int{1, 2, 3, 4}
    fmt.Println(arr)

    //值传递:拷贝实际参数arr的值给形式参数arr2
    update(arr)
    fmt.Println("调用函数后数据被修改为:", arr)
    //调用函数后,arr中的数据并未发生变化,说明是值传递
    //修改arr2的值并不会影响到arr的值
    //值传递:传递的是数据的副本,此时修改副本的数据,对原数据并不影响,究其根本在于二者并不在同一内存空间

    //引用传递
    //见下个案例
}

//值传递 操作的是数据本身 如:基础数据类型int、string、bool、floa64、array、struct...
//引用传递 操作的是数据的地址 如:slice、map、chan...

func update(arr2 [4]int) {
    fmt.Println("arr2接收的数据:", arr2)
    arr2[0] = 100
    fmt.Println("arr2修改后的数据:", arr2)
}

image-20240326224920045

案例3:引用传递,切片

package main

import "fmt"

func main() {
    //定义一个切片
    s1 := []int{1, 2, 3, 4}
    fmt.Println("s1默认数据", s1)
    update2(s1)
    fmt.Println("调用函数后s1的数据为:", s1)

}

//引用传递,传递的是数据的地址,当修改该数据时,由于和原数据共处同一地址处,原数据也会随着修改

func update2(s2 []int) {
    fmt.Println("s2 接收到的数据为:", s2)
    s2[0] = 10
    fmt.Println("s2[0]被修改后,s2的数据为:", s2)
}

image-20240326225156187

④函数作用域

变量的作用域是程序中变量可以正常访问的范围。在Go中:

  • 局部变量

    函数内定义的变量,它只在函数体内部或语句块内部可见。

  • 全局变量

    函数外部定义的变量。

案例1:内部与外部

package main

import "fmt"

func main() {
    x := "outside"
    {
        y := "inside"
        fmt.Println(y) // 输出 inside
    }
    fmt.Println(x) // 输出 outside
    // fmt.Println(y) // 编译错误:y在这里不可见
}

image-20240326225358932

案例2:局部变量,就近原则

package main

import "fmt"

var num int = 50

// 全局变量是 在函数外部定义的 变量
// 局部变量是 在函数内或语句块内定义的 变量 。它们只能在其定义的函数或语句块内部被访问。
func main() {

    //函数体内的局部变量
    temp := 100

    if b := 1; b <= 10 {
       //语句体内的局部变量
       temp := 50
       fmt.Println(temp) //局部变量 遵循就近原则

       fmt.Println(b)
    }
    fmt.Println(temp)

    fmt.Println(num)

    f1()
    f2()
}

func f1() {
    num := 40
    fmt.Println(num)
}

func f2() {
    fmt.Println(num)
}
⑤递归函数

递归函数是指一个函数在其定义中调用自己的函数。递归允许程序以简洁的方式解决复杂问题,在数据量小的时候,使用递过来实现是十分简便的,但是若数据量较大,不建议再使用递归函数来解决,因为这会占用太大的空间,程序效率较低。

因此,总结一下,使用递归时,一般需要注意以下几点:

  1. 确保有终止条件:递归函数必须有一个明确的终止条件,否则会无限递归下去,导致栈溢出错误。终止条件通常是一个或多个基案(base case),即最简单的问题实例,可以直接解答,不需要进一步递归。
  2. 注意栈空间的使用:每次函数调用时,都会在栈上为其分配空间来保存参数、局部变量和返回地址等信息。递归调用过深可能会耗尽栈空间,导致栈溢出。因此,对于可能导致深层递归的问题,需要谨慎设计递归逻辑,或考虑使用迭代等其他方法。
  3. 递归深度限制:在某些情况下,特别是处理大数据量或深度嵌套的数据结构时,考虑设置递归深度限制,以避免可能的栈溢出或过度消耗资源。

案例1:递归求阶乘

package main

import "fmt"

func factorial(n int) int {
    if n == 0 {
        return 1
    }
    return n * factorial(n-1)
}

func main() {
    fmt.Println(factorial(5)) // 输出 120
}

image-20240326230402126

案例2:递归求和

image-20230209215533739

package main

import "fmt"

func main() {
    sum := Sum(5)
    fmt.Println(sum)
}

// 递归函数 函数自己调用自己
// 递归需要有出口,也就是边界条件,否则会无限调用下去,导致栈溢出,系统崩溃
func Sum(n int) int {
    if n == 1 {
       return 1
    }
    return Sum(n-1) + n
}

image-20240326230529889

3、小结

  • 什么是函数?函数定义
  • 多个返回值的函数
  • 多个参数、可变参数 …
  • 参数的作用域( 小范围的可以用大范围的变量,反之则不行 )
  • 递归函数(函数自己调用自己,可能会导致栈溢出 … ,需要有终止条件或限制)

很感谢你能看到这里,如有相关疑问,还请下方评论留言。
Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊)
希望本篇内容能对大家有所帮助,如果大家喜欢的话,请动动手点个赞和关注吧,非常感谢你们的支持!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1547787.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Ubuntu】在Ubuntu中实现酣畅淋漓的性能释放:调整CPU频率

一、问题描述 在机器人开发中&#xff0c;经常需要运行诸如 SLAM 和 Planning 等 CPU 密集型程序&#xff0c;这些程序需要充分发挥计算机的性能&#xff0c;以确保算法的高效运行。然而&#xff0c;默认情况下&#xff0c;Ubuntu 通常将 CPU 设置为节能模式&#xff0c;导致 …

设计模式之组合模式解析

组合模式 1&#xff09;概述 1.定义 组合多个对象形成树形结构以表示具有“整体—部分”关系的层次结构。 组合模式对单个对象&#xff08;即叶子对象&#xff09;和组合对象&#xff08;即容器对象&#xff09;的使用具有一致性&#xff0c;组合模式又称为“整体—部分”(…

发送请求- header配置

请求头里是客户端的要求&#xff0c;把你的诉求告诉服务端&#xff0c;服务端按照你的要求返回数据 &#xff0c; 请求header需要严格全配置&#xff0c;把请求header全部传入&#xff0c;不能频繁访问&#xff0c;让后端知道它是正常请求 一般只配置User-Agent和Content Typ…

docker 搜索镜像命令

docker 搜索镜像命令 命令格式 docker search 关键字 如&#xff1a;docker centos 结果 result :

JVM(三)——字节码技术

三、字节码技术 1、类文件结构 一个简单的 HelloWorld.java package com.mysite.jvm.t5; // HelloWorld 示例 public class HelloWorld {public static void main(String[] args) {System.out.println("hello world");} }执行 javac -parameters -d . HellowWorld.…

零拷贝技术、常见实现方案、Kafka中的零拷贝技术的使用、Kafka为什么这么快

目录 1. 普通拷贝 2. 数据拷贝基础过程 2.1 仅CPU方式 2.2 CPU&DMA方式 3.普通模式数据交互 4. 零拷贝技术 4.1 出现原因 4.2 解决思路 4.2.1 mmap方式 4.2.2 sendfile方式 4.2.3 sendfileDMA收集 4.2.4 splice方式 5. Kafka中使用到的零拷贝技术 参考链接 本…

如何使用 ChatGPT 进行编码和编程

文章目录 一、初学者1.1 生成代码片段1.2 解释功能 二、自信的初学者2.1 修复错误2.2 完成部分代码 三、中级水平3.1 研究库3.2 改进旧代码 四、进阶水平4.1 比较示例代码4.2 编程语言之间的翻译 五、专业人士5.1 模拟 Linux 终端 总结 大多数程序员都知道&#xff0c;ChatGPT …

【二叉树】Leetcode 94. 二叉树的中序遍历【简单】

二叉树的中序遍历 给定一个二叉树的根节点 root &#xff0c;返回 它的 中序 遍历 。 示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&#xff1a;[1,3,2] 解题思路 中序遍历是一种二叉树遍历方式&#xff0c;按照“左根右”的顺序遍历二叉树节点。 1、递归…

基于单片机的智能汽车防盗系统设计

摘要:本文介绍了一种以汽车专用单片机飞思卡尔MC68HC908QT4CPE 作为底层主控芯片,人体远红外热释传感器、防闯入光幕墙及振动传感器作为检测装置的汽车防盗系统。单片机将检测到的防盗传感器开关信号,通过数据分析,系统可以设置布防和解除布防模式,在布防模式下,当检测到…

SSH配置公钥私钥免密登录——windows to linux

SSH配置公钥私钥免密登录——windows to linux SSH的安全机制一、修改远程主机ssh设置二、在windows客户端生成公钥私钥文件三、将客户端公钥追加到远程主机 .ssh/authorized_keys中参考链接 SSH的安全机制 SSH之所以能够保证安全&#xff0c;原因在于它采用了非对称加密技术(…

MATLAB 自定义生成圆柱点云(49)

MATLAB 自定义生成圆柱点云(49) 一、算法介绍二、具体实现1.代码2.效果一、算法介绍 按照一些提前指定的圆柱参数,自定义生成圆柱点云,可添加噪声,用于后续的实验测试 二、具体实现 1.代码 代码如下(示例): % 指定圆柱的参数 radius = 5; % 圆柱半径 height = 20…

【unity】如何汉化unity Hub

相信大家下载安装unity后看着满操作栏的英文&#xff0c;英文不好的小伙伴们会一头雾水。但是没关系你要记住你要怎么高速运转的机器进入中国&#xff0c;请记住我给出的原理&#xff0c;不懂不代表不会用啊。现在我们就来把编译器给进行汉化。 第一步&#xff1a;我们打开Uni…

Spring Boot | Spring Boot的“核心配置“与“注解“

目录: Spring Boot的核心配置与注解 &#xff1a;1. 全局配置文件 ( application.properties / application.yaml&#xff1a;创建项目时候自动生成&#xff0c;其会被“自动导入”到“程序”中 )application.properties配置文件application.yaml 配置文件 (推荐使用)当value值…

模板设计模式经典案例

模板设计模式讲究的是将不变的设置为基类&#xff0c;将变的设置为虚函数来让子类实现。下面就以这样的写下模板设计模式的例子。 例子场景 一个工程步骤分为step1,step2&#xff0c;其中step1由总工程指定&#xff0c;step2由子工程指定&#xff0c;最后由一个函数串起来&am…

Android TargetSdkVersion 30 安装失败 resources.arsc 需要对齐且不压缩。

公司项目&#xff0c;之前targetSDKVersion一直是29&#xff0c;近期小米平台上架强制要求升到30&#xff0c;但是这个版本在android12上安装失败&#xff0c;我用adb命令安装&#xff0c;报错如下图 adb: failed to install c: Program Files (x86)(0A_knight\MorkSpace \Home…

SpringDoc 注解

列举几个常用的 1. Tag 用于说明或定义的标签。一般作用于控制层 2.Operation(summary "这是新增方法") 描述 API 操作的元数据信息。常用于 controller 层的方法上 ​ 3.Parameter 用于描述 API 操作中的参数 ​ 4.Operation Parameters ​ 5.Schema用于…

R语言实现——网状 Meta 分析

近来年&#xff0c;网状 Meta 分析相关研究不断涌现&#xff0c;此类研究不但能发表在国内各大核心期刊上&#xff0c;还能在SCI期刊甚至医学4大刊上看到其身影。随手在pubmed上面一搜索&#xff0c;就能得到一万多篇相关文献。俨然成为医学文献研究的“大杀器”&#xff01; P…

智慧公厕,让数据和技术更好服务社会生活

智慧公厕&#xff0c;作为智慧城市建设中不可忽视的一部分&#xff0c;正逐渐受到越来越多人的关注。随着科技的不断进步&#xff0c;智能化公厕已经成为一种趋势&#xff0c;通过数据的流转和技术的整合&#xff0c;为社会生活带来了更好的服务。本文以智慧公厕源头实力厂家广…

selenium元素定位--xpath定位--层级与逻辑组合定位

其他元素非唯一时&#xff0c;又不想用xpath绝对定位时&#xff0c;需要用到层级与逻辑定位. 一、层级属性结合定位&#xff1a; 遇到元素没有class、name、id等或属性动态变化情况时&#xff0c;可以找父节点元素&#xff0c;父级节点没有id时&#xff0c;可以继续往上找id&…

Flutter(踩坑)之Android sdkmanager tool not found

D:\Flutter\flutter\bin\flutter.bat doctor --verbose [√] Flutter (Channel stable, v1.2.1, on Microsoft Windows [Version 10.0.22631.3296], locale zh-CN)• Flutter version 1.2.1 at D:\Flutter\flutter• Framework revision 8661d8aecd (5 years ago), 2019-02-14 …