草庐IT

【Golang】案例为基浅谈Go的变量与常量

桌子椅子凳子。 2023-07-09 原文

📓推荐网站(不断完善中):个人博客

📌个人主页:个人主页

👉相关专栏:CSDN专栏个人专栏

🏝立志赚钱,干活想躺,瞎分享的摸鱼工程师一枚

🍒前言

在上一篇文章中,博主对于Go语言中的一些基本语法与变量的初步定义做了初步介绍。

在这篇文章主要是对于变量的进一步应用,以及对于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,那么你可以发现在改变初始值的情况下,后续的值也会陆续发生改变,可以动手试试哦!


5.关于源码

博主将接下来系列中学习到的源码做了统一保存,在 Github 上,分享给有需要的人(禁止未经说明直接复制转载)。

如有需要可以 点击这里🚪 进行查看,创作不易,如果您觉得不错可以给博主点个 star🌟 ,感激不尽!

希望能够通过 案例 + 注释 + 讲解 的方式在学习过程中的经验和理解分享给大家同时能对想要学习的人提供一些帮助。

🍊写在结尾

本章节内容主要是讲述了一些关于变量的进一步细节运用,以及关于常量的知识,这两部分的内容在未来我们实际开发代码当中还是比较经常用到的,所以还是要好好的去梳理和理解一下。

以上为个人分享,如有错误请指出

有关【Golang】案例为基浅谈Go的变量与常量的更多相关文章

  1. ruby-on-rails - 未初始化的常量 Psych::Syck (NameError) - 2

    在我的gem中,我需要yaml并且在我的本地计算机上运行良好。但是在将我的gem推送到ruby​​gems.org之后,当我尝试使用我的gem时,我收到一条错误消息=>"uninitializedconstantPsych::Syck(NameError)"谁能帮我解决这个问题?附言RubyVersion=>ruby1.9.2,GemVersion=>1.6.2,Bundlerversion=>1.0.15 最佳答案 经过几个小时的研究,我发现=>“YAML使用未维护的Syck库,而Psych使用现代的LibYAML”因此,为了解决

  2. ruby-on-rails - 如何使用 instance_variable_set 正确设置实例变量? - 2

    我正在查看instance_variable_set的文档并看到给出的示例代码是这样做的:obj.instance_variable_set(:@instnc_var,"valuefortheinstancevariable")然后允许您在类的任何实例方法中以@instnc_var的形式访问该变量。我想知道为什么在@instnc_var之前需要一个冒号:。冒号有什么作用? 最佳答案 我的第一直觉是告诉你不要使用instance_variable_set除非你真的知道你用它做什么。它本质上是一种元编程工具或绕过实例变量可见性的黑客攻击

  3. ruby - 通过 ruby​​ 进程共享变量 - 2

    我正在编写一个gem,我必须在其中fork两个启动两个webrick服务器的进程。我想通过基类的类方法启动这个服务器,因为应该只有这两个服务器在运行,而不是多个。在运行时,我想调用这两个服务器上的一些方法来更改变量。我的问题是,我无法通过基类的类方法访问fork的实例变量。此外,我不能在我的基类中使用线程,因为在幕后我正在使用另一个不是线程安全的库。所以我必须将每个服务器派生到它自己的进程。我用类变量试过了,比如@@server。但是当我试图通过基类访问这个变量时,它是nil。我读到在Ruby中不可能在分支之间共享类变量,对吗?那么,还有其他解决办法吗?我考虑过使用单例,但我不确定这是

  4. ruby-on-rails - active_admin 目录中的常量警告重新声明 - 2

    我正在使用active_admin,我在Rails3应用程序的应用程序中有一个目录管理,其中包含模型和页面的声明。时不时地我也有一个类,当那个类有一个常量时,就像这样:classFooBAR="bar"end然后,我在每个必须在我的Rails应用程序中重新加载一些代码的请求中收到此警告:/Users/pupeno/helloworld/app/admin/billing.rb:12:warning:alreadyinitializedconstantBAR知道发生了什么以及如何避免这些警告吗? 最佳答案 在纯Ruby中:classA

  5. ruby-on-rails - 如何在我的 Rails 应用程序 View 中打印 ruby​​ 变量的内容? - 2

    我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby​​中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R

  6. ruby - RuntimeError(自动加载常量 Apps 多线程时检测到循环依赖 - 2

    我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("

  7. ruby-on-rails - ActionController::RoutingError: 未初始化常量 Api::V1::ApiController - 2

    我有用于控制用户任务的Rails5API项目,我有以下错误,但并非总是针对相同的Controller和路由。ActionController::RoutingError:uninitializedconstantApi::V1::ApiController我向您描述了一些我的项目,以更详细地解释错误。应用结构路线scopemodule:'api'donamespace:v1do#=>Loginroutesscopemodule:'login'domatch'login',to:'sessions#login',as:'login',via::postend#=>Teamroutessc

  8. ruby - 获取模块中定义的所有常量的值 - 2

    我想获取模块中定义的所有常量的值:moduleLettersA='apple'.freezeB='boy'.freezeendconstants给了我常量的名字:Letters.constants(false)#=>[:A,:B]如何获取它们的值的数组,即["apple","boy"]? 最佳答案 为了做到这一点,请使用mapLetters.constants(false).map&Letters.method(:const_get)这将返回["a","b"]第二种方式:Letters.constants(false).map{|c

  9. ruby - 在 Ruby 中重新分配常量时抛出异常? - 2

    我早就知道Ruby中的“常量”(即大写的变量名)不是真正常量。与其他编程语言一样,对对象的引用是唯一存储在变量/常量中的东西。(侧边栏:Ruby确实具有“卡住”引用对象不被修改的功能,据我所知,许多其他语言都没有提供这种功能。)所以这是我的问题:当您将一个值重新分配给常量时,您会收到如下警告:>>FOO='bar'=>"bar">>FOO='baz'(irb):2:warning:alreadyinitializedconstantFOO=>"baz"有没有办法强制Ruby抛出异常而不是打印警告?很难弄清楚为什么有时会发生重新分配。 最佳答案

  10. ruby-on-rails - 使用 ruby​​ 将多个实例变量转换为散列的更好方法? - 2

    我收到格式为的回复#我需要将其转换为哈希值(针对活跃商家)。目前我正在遍历变量并执行此操作:response.instance_variables.eachdo|r|my_hash.merge!(r.to_s.delete("@").intern=>response.instance_eval(r.to_s.delete("@")))end这有效,它将生成{:first="charlie",:last=>"kelly"},但它似乎有点hacky和不稳定。有更好的方法吗?编辑:我刚刚意识到我可以使用instance_variable_get作为该等式的第二部分,但这仍然是主要问题。

随机推荐