变量的作用域是指程序代码中能够有效使用这个变量的范围。

不要将作用域和生命期混在一起。

作用域是代码中的一块区域,是一个编译期的属性;生命期是程序执行期间变量存活的时间段。在此时间段内,变量能够被程序的其他部分所引用,是执行期的概念。

语法块是包括在花括号内的一系列语句。比如函数体或者循环体。语法块内部声明的变量是无法被语法块外部代码訪问的。我们能够扩展局部语法块的概念,在某些场景下。是不须要花括号的,这样的形式称之为词法块。

词法块分为几种:全局词法块,包括全部源码。包 词法块。包括整个package;文件词法块。包括整个文件;for、if、switch语句的词法块;switch或select中的case分支的词法块。当然也包括之前提到的语法块

声明语句的词法块决定了变量的作用域。

Go语言的内置类型、内置函数、内置常量都是全局词法块,因此它们都是全局作用域的。比如int、len、true等。能够在整个程序直接使用;对于导入的包。比如temconv导入的fmt包,是文件词法块,因此仅仅能在当前文件里訪问fmt包。这里fmt是全文件范围的作用域;tempconv.CToF函数中的变量c,则是局部词法块(语法块)的,因此它的作用域是函数的内部。

控制语句后面的标签(label)。比如break、continue或goto后的标签。它们的作用域是在控制语句所在的函数内部。

一个程序可能会有多个同样的变量名,仅仅要它们的声明在不同的词法块就好。

比如。你能够在函数内声明一个局部变量x,同一时候再声明一个包级的变量x,这是在函数内部,局部变量x就会替代后者,这里称之为shadow,意味着在函数作用域内局部变量将包变量隐藏了。

当编译器遇到一个变量名的引用时,会去搜索该变量的声明语句,首先从最内部的词法块開始,然后直到全局词法块。假设编译期找不到变量名的声明语句。那么就会报错:undeclared name。假设变量名在内部的词法块和外部的词法块同一时候声明,那么依据编译期的搜索规则,内部的词法块会先找到。

在这样的情况下。内部的声明会隐藏外部的声明(shadow),此时。外部声明的变量是无法訪问的:

func f() {}

var g = "g"

func main() {
f := "f"
fmt.Println(f) // "f"; 本地变量f隐藏了包级函数f
fmt.Println(g) // "g"; 包级变量g
fmt.Println(h) // compile error: undefined: h
}

在函数内部。词法块能够嵌套随意层数,因此本地变量能够隐藏外部变量。大多数的语法块(花括号)是通过控制语句if、for等创建的,以下的程序有三个不同的x变量。每一个变量都是声明在不同的词法块中(这段代码主要是为了说明作用域的规则,这样的编码风格并不提倡!):

func main() {
x := "hello!"
for i := 0; i < len(x); i++ {
x := x[i]
if x != '!' {
x := x + 'A' - 'a'
fmt.Printf("%c", x) // "HELLO" (one letter per iteration)
}
}
}

表达式x[i]和x + 'A' - 'a' 分别引用了不同的x变量,后面会解释。

就像之前提到的那样,不是全部的词法块都有显式的花括号。上面的for循环创建了两个词法块:带花括号的循环主体,显式词法块;还有不带花括号的隐式词法块。比如for循环的条件语句中声明一个变量i。这里i的作用域包括for的条件语句和for的主体。

以下的样例也创建了三个变量x,每一个都在不同的词法块中声明,一个在函数主体中,一个在for的隐式词法块中,另一个在显式词法块-循环主体中,这当中仅仅有1和3是显式词法块:

func main() {
x := "hello"
for _, x := range x {
x := x + 'A' - 'a'
fmt.Printf("%c", x) // "HELLO" (每次循环一个字符)
}
}

就像for循环一样。if语句和switch语句一样会创建隐式词法块。以下的代码在if-else链中说明了x和y的作用域:

if x := f(); x == 0 {
fmt.Println(x)
} else if y := g(x); x == y {
fmt.Println(x, y)
} else {
fmt.Println(x, y)
}
fmt.Println(x, y) // compile error: x and y are not visible here

第二个if语句嵌套在第一个里面。所以第一个if语句里声明的变量对第二个if语句是可见的。在switch中也有相似的规则:除了条件词法块外。每一个case也有自己的词法块。

对于包级变量来说,声明的顺序和作用域是无关的,全部一个包级变量声明时能够引用它自身也能够引用在它之后声明的包级变量。然而,假设一个变量或者常量在声明时引用了它自己。编译器会报错。

看以下的程序:

if f, err := os.Open(fname); err != nil { // compile error: unused: f
return err
}
f.ReadByte() // compile error: undefined f
f.Close() // compile error: undefined f

f的作用域仅仅是if语句,因此在if之外的词法块是不可訪问的,报编译错误。

这里能够更改代码,提前声明f变量:

f, err := os.Open(fname)
if err != nil {
return err
}
f.ReadByte()
f.Close()

假设不想在外部词法块声明变量,能够这么写:

if f, err := os.Open(fname); err != nil {
return err
} else {
// f and err are visible here too
f.ReadByte()
f.Close()
}

可是第三种不是Go推荐的写法。另外一种比較合适,将正常逻辑和错误处理分离。

短声明变量的作用域是要特别注意的,考虑以下的程序,開始时会获取当前的工作文件夹,保存在一个包级变量中。

这个本来能够通过在main函数中调用os.Getwd来完毕,可是用init函数将这块儿逻辑从主逻辑中分离是一个更好的选择,特别是由于获取文件夹的操作可能会是失败的,这个时候须要处理返回的错误。函数log.Fatalf会打印一条信息。然后调用os.Exit(1)终结程序:

var cwd string

func init() {
cwd, err := os.Getwd() // compile error: unused: cwd
if err != nil {
log.Fatalf("os.Getwd failed: %v", err)
}
}

这里cwd和err在init的词法块中都没有声明过,因此 := 语句会将它们两声明为本地变量。init内部的cwd声明会隐藏外部的,因此这个程序没有达到更新包级变量cwd的目的。

当前版本号,Go编译器会检測到本地变量cwd从未使用,因此会报错,可是这样的检查并非非常严格。比如。假设在log.Fatalf中打印cwd的值(这时本地变量cwd会被使用)。那么这样的错误就会被隐藏!!

var cwd string

func init() {
cwd, err := os.Getwd() // 注意这里的包级cwd被本地隐藏了,可是编译器没有报错!
if err != nil {
log.Fatalf("os.Getwd failed: %v", err)
}
log.Printf("Working directory = %s", cwd)
}

这里全局变量cwd没有得到正确的初始化,同一时候,log函数使用了cwd本地变量。隐藏了这个bug。

有一些办法能够处理这样的潜在的错误,最直接的就是避免使用:=,通过var来声明err变量:

var cwd string

func init() {
var err error
cwd, err = os.Getwd()
if err != nil {
log.Fatalf("os.Getwd failed: %v", err)
}
}

在这一章里,我们简单学习了包。文件。声明。语句等。

在接下来的两章,我们会学习数据结构

PS. 这一章真心非常难写,足足用了3个小时。作为质量对照大家能够參见这篇文章Scope

文章全部权:Golang隐修会 联系人:孙飞。CTO@188.com!

Go语言核心之美 1.5-作用域的更多相关文章

  1. Go语言核心之美-必读

    Go语言核心之美开篇了!.不管你是新手还是一代高人,在这个系列文章中.总能找到你想要的! 博主是计算机领域资深专家并且是英语专8水平,翻译标准仅仅有三个:精确.专业.不晦涩,为此每篇文章可能都要耗费数 ...

  2. [程序设计语言]-[核心概念]-02:名字、作用域和约束(Bindings)

    本系列导航 本系列其他文章目录请戳这里. 1.名字.约束时间(Binding Time) 在本篇博文开始前先介绍两个约定:第一个是“对象”,除非在介绍面向对象语言时,本系列中出现的对象均是指任何可以有 ...

  3. Go语言核心之美 3.2-slice切片

    Slice(切片)是长度可变的元素序列(与之相应,上一节中的数组是不可变的),每一个元素都有同样的类型.slice类型写作[]T.T是元素类型.slice和数组写法非常像,差别在于slice没有指定长 ...

  4. Go语言核心之美 1.1-命名篇

    命名篇 1.Go的函数.变量.常量.自己定义类型.包(Package)的命名方式遵循以下规则: 1)首字符能够是随意的Unicode字符或者下划线 2)剩余字符能够是Unicode字符.下划线.数字 ...

  5. Go语言核心之美 4.3-多返回值

    在Go语言中.函数能够有多个返回值,这个特性我们已经在之前的样例见过非常多,非常多标准库函数都会返回两个值,一个是期望得到的函数执行结果,另外一个是函数出错时的错误值. 以下的程序是findlinks ...

  6. JS 语言核心(JavaScript权威指南第六版)(阅读笔记)

    前言: 对于程序员,学习是无止境的,知识淘换非常快,能够快速稳固掌握一门新技术,是一个程序员应该具备的素质.这里将分享本人一点点不成熟的心得. 了解一门语言,了解它的概念非常重要,但是一些优秀的设计思 ...

  7. 第一部分 JavaScript语言核心(二)

    第四章 表达式和运算符 P66 运算符优先级,从上到下: p68 属性访问表达式和调用表达式的优先级比运算符优先级都要高,eg: typeof my.functions[x](y) //typeof在 ...

  8. 第一章:Javascript语言核心

    本节是javascript语言的一个快速预览,也是本书的第一部分快速预览. 读此书之前,感谢淘宝技术团队对此javascript核心的翻译,感谢弗拉纳根写出此书.感谢你们无私的分享,仅以此笔记献给你们 ...

  9. [java学习笔记]java语言核心----面向对象之this关键字

    一.this关键字 体现:当成员变量和函数的局部变量重名时,可以使用this关键字来区别:在构造函数中调用其它构造函数 原理:         代表的是当前对象.         this就是所在函数 ...

随机推荐

  1. PostgreSQL创建只读用户

    创建用户及指定密码: CREATE USER readonly WITH ENCRYPTED PASSWORD 'ropass'; 设置用户默认事务只读: alter user readonly se ...

  2. python-排序算法 冒泡和快速排序

    交换排序 交换排序有冒泡排序和快速排序 冒泡排序 冒泡排序就是每次找出最大(最小)元素,放在集合最前或最后,这是最简单的排序算法 print("未排序之前:",collection ...

  3. FZOJ 2176 easy problem ( 树链剖分 )

    pid=2176" target="_blank">题目链接~~> 做题感悟:感觉做多了树链剖分的题目,有很多是树链剖分 + 想法.. 解题思路: 这题非常明 ...

  4. Linux下无需输入password自己主动登陆sshserver方法

    用OpenSSH在linux下登陆sshserver时.每次都提示要输入password,并且使用vim 的netrw插件编辑远程文件时每次改动后保存都要输password,很麻烦. 查看了netrw ...

  5. CO-PRIME(初探 莫比乌斯)NYOJ1066(经典)gcd(a,b)=1

    CO-PRIME 时间限制:1000 ms  |  内存限制:65535 KB 难度:3 描写叙述 This problem is so easy! Can you solve it? You are ...

  6. leetcode第一刷_Text Justification

    这个题的接受率好低,搞得我一直不敢做.后来认真的看了一下题目,不是非常难嘛.字符串的题目ac率就是低,除了难,还由于它的測试用例太多. 思路不难,主要是由于特殊情况太多.纯模拟,我把全部的情况罗列一下 ...

  7. 洛谷P1919 【模板】A*B Problem升级版(FFT快速傅里叶)

    题目描述 给出两个n位10进制整数x和y,你需要计算x*y. 输入输出格式 输入格式: 第一行一个正整数n. 第二行描述一个位数为n的正整数x. 第三行描述一个位数为n的正整数y. 输出格式: 输出一 ...

  8. 《五》uploadify插件上传文件

    下载地址:http://www.uploadify.com/wp-content/uploads/files/uploadify.zip 相关配置:http://www.uploadify.com/d ...

  9. VUE里子组件获取父组件动态变化的值

    在VUE里父组件给子组件间使用props方式传递数据,但是希望父组件的一个状态值改变然后子组件也能监听到这个数据的改变来更新子组件的状态. 场景:子组件通过props获取父组件传过来的数据,子组件存在 ...

  10. tensorflow学习之路-----MNIST数据

    ''' 神经网络的过程:1.准备相应的数据库 2.定义输入成 3.定义输出层 4.定义隐藏层 5.训练(根据误差进行训练) 6.对结果进行精确度评估 ''' import tensorflow as ...