通过案例详细聊聊Go语言的变量与常量

 更新时间:2023年03月17日 09:21:41   作者:桌子椅子凳子。  
在任何一门现代的高级语言中,变量和常量都是它非常基础的程序结构的组成部分,下面这篇文章主要给大家介绍了关于如何通过案例详细聊聊Go语言的变量与常量的相关资料,需要的朋友可以参考下

1.变量交换

简单阐述一下什么是变量交换,顾名思义,就是将变量的值进行交换的一个动作

例如我们定义了两个变量a:=10b:=20,那么这时候我们想要将两个变量的值进行交换的话,在Go语言中为我们提供了比较方便的方式。

案例

package main

import "fmt"

func main() {
	
	// 定义变量a、b,并且分别赋值10、20
	a, b := 10, 20
	// 第一轮输出查看结果
	fmt.Println("原本a、b的结果为:", a, b)
	// 然后我们可以直接用 = 号来对a、b进行快速赋值,交换他们的位置
	a, b = b, a
  // 第二轮输出查看结果
	fmt.Println("新输出的a、b结果为:", a, b)

}

通过以上代码我们可以看到,在=的右侧寓意为赋值,因为在Go中支持多赋值和多声明的代码形式,所以我们只需要将要赋值的变量进行位置对换,那么新的值就被赋予了。

2.匿名变量

阐述完变量交换,那么我们就不得不提变量中的一个特殊存在,那就是匿名变量

将值赋予给匿名变量实际上也就等于将这个值给抛弃了。通常也称之为抛弃值

关键标识

利用 _ 下划线,来标识匿名变量,也是我们所说的空白标识符

案例

package main

import "fmt"

func main() {

	// 定义变量
	_, a := 10, 7
	// 使用匿名变量的时候10就是抛弃值
	fmt.Println("输出结果:", a)
}

2.1.如何理解这个匿名变量的使用场景?

可能很多人会有疑问这个匿名变量有什么意义,其实在实际的使用场景中,匿名变量的使用意义个人感觉更趋近于占位符。

假设我们用到了某个函数名字叫Dial()的函数,这个函数会返回两个值给我们,那么我们只需要其中的一个值即可,另一个值我们用不到,但是因为是返回值,代码规则里面我们要接收,所以这时候我们就可以用匿名变量来占住另一个值的位置,代码如下。

案例

package main

import (
	"fmt"
	"net"
)
func main() {

    //conn, err := net.Dial("tcp", "127.0.0.1:8080")
    //如果不想接收err的值,那么可以使用_表示,这就是匿名变量
    conn, _ := net.Dial("tcp", "127.0.0.1:8080")
		fmt.Println(conn)
}

这里的net.Dial()函数就有两个返回值,分别为conn,err,那么我们只需要conn这个返回值的时候,我们就可以利用到匿名变量来占位获取到另一个返回值,但是我们可以不需要用到这个返回值。

3.关于作用域

一个变量(常量、类型或函数)在程序中都有一定的作用范围,称之为作用域

首先Go语言的特殊性导致在编译的过程中如果出现你定义的变量、常量、函数等没有被使用的话,编译是无法通过的,所以了解作用范围是很有必要的。

了解作用域可以更好的定义你需要使用的内容,也可以解决一些编译过程中的错误。

根据定义的位置总结三种

  • 函数内定义的变量称为局部变量
  • 函数外定义的变量称为全局变量
  • 函数定义中的变量称为形式参数

所谓的函数可以暂时先理解成一块代码片段,比如我们案例中常用的main()就是函数

将内容定义在函数的{ }内,就可以称之为定义在函数体内。

3.1.局部变量

在函数体内声明的变量称之为局部变量,它们的作用域只在函数体内,函数的参数和返回值变量都属于局部变量。

(扩展)局部变量不是一直存在的,它只在定义它的函数被调用后存在,函数调用结束后这个局部变量就会被销毁。

案例

package main

import "fmt"

func main() {

	// 定义变量,以下这些定义在main()函数中的都可以称之为局部变量
	a := 10
	b := 20
	c := a + b
  // 输出局部变量
	fmt.Println("输出局部变量为:", a, b, c)

}

3.2.全局变量

在函数体外声明的变量称之为全局变量,全局变量只需要在一个源文件中定义,就可以在所有源文件中使用,当然,不包含这个全局变量的源文件需要使用“import”关键字引入全局变量所在的源文件之后才能使用这个全局变量。

全局变量声明必须以 var 关键字开头,如果想要在外部包中使用(也就是想要被别的包进行import引用),那么全局变量的首字母必须大写。

案例

package main

import "fmt"

// 定义全局变量,必须有var关键字,并且在函数体外面
var a = 10
var b = "let's go"

func main() {
	// 在main()函数中调用a、b
	fmt.Println("输出全局变量为:", a, b)
	demo()
}

func demo() {
	// 因为是全局使用,在要在这个源码中我们的方法都能调用到
	// 因此我在sum()函数中也可以调用到b变量的值
	fmt.Println("调用sum()函数输出的全局变量为:", b)
}

可以看到如上案例,无论是在main()函数中还是在我们的demo()函数中,我们都可以得到a、b的值进行输出

使用细节:关于全局与局部同名怎么办?

大家可能会思考一个问题,如果我有一个全局变量叫a,同时我又在函数中也定义了一个局部变量a,这两个变量的名字相同但是值不同,那么我们程序应该采用谁?

其实大部分时候遇到这种情况,程序当中遵循的一般都是就近原则,也就是说很多时候我们的程序是谁离得近,就用谁。

也就说出现如上情况我们会优先先采用离得近的局部变量

案例

package main

import "fmt"

var a = "这是全局变量"

func main() {

	a := "这是局部变量"
	fmt.Println("在main中会采用的变量是:", a)
	demo()
}

func demo() {
	fmt.Println("如果没有局部变量影响: ", a)
}

得到的结果是:

在main中会采用的变量是: 这是局部变量
如果没有局部变量影响: 这是全局变量

注意:虽然这种情况会出现,但是还是不要出现这种让人容易产生歧义的代码,规则虽然没有不允许但是代码规范上显然这样是不对的

3.3.形式参数

在定义函数时函数名后面括号中的变量叫做形式参数(简称形参)。形式参数只在函数调用时才会生效,函数调用结束后就会被销毁,在函数未被调用时,函数的形参并不占用实际的存储单元,也没有实际值。

形式参数某种意义上也是一个被声明的局部变量,所以我们在定义局部变量的时候不要出现局部变量与形式参数重名的情况为好。

案例

package main

import "fmt"

// 定义了全局变量
var a = "这是全局变量"

func main() {
	// 这里是局部变量
	a := "这是main的局部变量a"
	b := "这是main的局部变量b"
	fmt.Println("输出变量的值为: ", a, b)
	// 在调用test()函数的时候传递局部变量a、b的值
	test(a, b)
}

// 在函数中定义形参a、b,那么当test()函数被调用的时候
// 形参就可以直接被当作局部变量来使用,值则是由调用者的传递内容来决定的
func test(a, b string) {
	// 输出结果
	fmt.Println("这是由main传过来的内容:", a, b)
}

得到的输出结果为:

输出变量的值为: 这是main的局部变量a 这是main的局部变量b
这是由main传过来的内容: 这是main的局部变量a 这是main的局部变量b

通过上述案例,让我们对 局部变量 、 全局变量 、 形式参数 有一个了解,并且分清楚他们之间的区别和使用

4.关于常量

在我们开始介绍代码中的常量的时候,我们要先阐述一下什么叫做常量?

实际上所谓的常量就是一个不会改变的值,用来存储一些我们自己事先规定好、不会改变的值

4.1.常量的使用

Go语言中的常量使用关键字const定义,用于存储不会改变的数据,常量是在编译时被创建的,即使定义在函数内部也是如此,并且只能是布尔型、数字型(整数型、浮点型和复数)和字符串型。

由于编译时的限制,定义常量的表达式必须为能被编译器求值的常量表达式

语法

// 注意了 []表示 数据类型 可写,可不写
const 常量名 [数据类型] = 值

从之前的变量声明中我们可以知道,其实Go语言很多时候想要简化一些批量性的动作,因此在定义常量的时候也可以采用批量式的定义。

批量定义的语法

const (
	常量名1 = 值1
  常量名2 = 值2
  ...
)

所有常量的运算都可以在编译期完成,这样不仅可以减少运行时的工作,也方便其他代码的编译优化,当操作数是常量时,一些运行时的错误也可以在编译时被发现,例如整数除零、字符串索引越界、任何导致无效浮点数的操作等。

案例

// 单个定义常量
const pi= 3.14159

// 批量定义常量
const (
    e  = 2.7182818
    pi = 3.1415926
)

4.2.常量使用的细节

其实在使用常量的过程中,我们除了采用批量定义的方式来定义常量,我们还可以用另一种方式在批量定义常量的同时,简化常量初始化(赋值)的过程。

简单来说,假设我要定义a、b、c、d四个常量。a、b的值都定义为10,c、d的值都定义为20,那么我们在初始化的时候就可以进行一些简化操作。

案例解析

package main

import "fmt"

const (
	a = 10
	b
	c = 20
	d
)

func main() {

	fmt.Println("输出定义的常量:", a, b, c, d)
}

最终得到的输出结果为:

输出定义的常量: 10 10 20 20

4.3.iota常量生成器

常量声明可以使用 iota 常量生成器初始化,它用于生成一组以相似规则初始化的常量,但是不用每行都写一遍初始化表达式。

简单来说呢,就是有点类似于计数器的存在,在定义第一个常量的时候会被归置为0,然后依此累加1

需求

我们从需求出发可能更好理解这个iota

假设我们要定义常量为周一到周天,那么周一为0,依次类推周天为6。

案例解析

package main

import "fmt"

// 定义周一到周天
const (
	Monday = iota
	Tuesday
	Wednesday
	Thursday
	Friday
	Saturday
	Sunday //6
)

func main() {

	fmt.Println("输出日期的值为:", Monday, Tuesday, Wednesday, Thursday, Friday, Saturday, Sunday)
}

输出的结果为:

输出日期的值为: 0 1 2 3 4 5 6

当然我们知道iota的初始值默认归0,但是只要你在第一步对iota进行运算操作,比如Monday = iota+1,那么你可以发现在改变初始值的情况下,后续的值也会陆续发生改变,可以动手试试哦!

总结

到此这篇关于Go语言的变量与常量的文章就介绍到这了,更多相关Go变量与常量内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Golang中如何对MySQL进行操作详解

    Golang中如何对MySQL进行操作详解

    这篇文章主要给大家介绍了关于在Golang中如何对MySQL进行操作的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者使用Golang具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-03-03
  • golang1.16新特性速览(推荐)

    golang1.16新特性速览(推荐)

    这篇文章主要介绍了golang1.16新特性速览,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-02-02
  • 详解Golang中Requests包的使用

    详解Golang中Requests包的使用

    Go的net/http包虽然功能强大、用途也广告,但要想正确的使用请求的客户端是非常繁琐的,所以本文和大家分享一个高效的HTTP的请求包carlmjohnson/requests的使用,需要的小伙伴可以了解一下
    2023-06-06
  • 深入探索Go语言中的高效数据结构堆

    深入探索Go语言中的高效数据结构堆

    堆,作为一种基本的数据结构,以其在优先队列和排序算法中提供高效解决方案的能力而闻名。在本文中,我们将深入探讨堆的内部工作原理,包括其特性、实现细节以及在现代编程中的应用
    2008-06-06
  • golang 对私有函数进行单元测试的实例

    golang 对私有函数进行单元测试的实例

    这篇文章主要介绍了golang 对私有函数进行单元测试的实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-05-05
  • 详解go中panic源码解读

    详解go中panic源码解读

    这篇文章主要介绍了go中panic源码解读,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-04-04
  • Golang实现AES对称加密算法实例详解

    Golang实现AES对称加密算法实例详解

    所谓对称加密是指在加密和解码时使用同一密钥的加密方式,下面这篇文章主要给大家介绍了关于Golang实现AES对称加密算法的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-02-02
  • 详解Golang如何优雅的终止一个服务

    详解Golang如何优雅的终止一个服务

    后端服务通常会需要创建子协程来进行相应的作业,但进程接受到终止信号或正常结束时,并没有判断或等待子协程执行结束,下面这篇文章主要给大家介绍了关于Golang如何优雅的终止一个服务的相关资料,需要的朋友可以参考下
    2022-03-03
  • GoFrame框架gset交差并补集使用实例

    GoFrame框架gset交差并补集使用实例

    这篇文章主要为大家介绍了GoFrame框架gset交差并补集使用实例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • 深入了解Go语言中database/sql是如何设计的

    深入了解Go语言中database/sql是如何设计的

    在 Go 语言中内置了 database/sql 包,它只对外暴露了一套统一的编程接口,便可以操作不同数据库,那么database/sql 是如何设计的呢,下面就来和大家简单聊聊吧
    2023-07-07

最新评论