每道题都有亲自测试,有不明白的可以私信我。

选择题

1. 【初级】下面属于关键字的是()
A. func
B. def
C. struct
D. class
参考答案:AC

2. 【初级】定义一个包内全局字符串变量,下面语法正确的是()
A. var str string
B. str := ""
C. str = ""
D. var str = ""
参考答案:AD

简短形式,使用 := 赋值操作符,a 和 b 的类型(int 和 bool)将由编译器自动推断。这是使用变量的首选形式,但是它只能被用在函数体内,而不可以用于全局变量的声明与赋值。使用操作符 := 可以高效地创建一个新的变量,称之为初始化声明。


3. 【初级】通过指针变量 p 访问其成员变量 name,下面语法正确的是()
A. p.name
B. (*p).name
C. (&p).name
D. p->name


参考答案:AB
4. 【初级】关于接口和类的说法,下面说法正确的是()
A. 一个类只需要实现了接口要求的所有函数,我们就说这个类实现了该接口
B. 实现类的时候,只需要关心自己应该提供哪些方法,不用再纠结接口需要拆得多细才合理
C. 类实现接口时,需要导入接口所在的包
D. 接口由使用方按自身需求来定义,使用方无需关心是否有其他模块定义过类似的接口

参考答案:ABD


5. 【初级】关于字符串连接,下面语法正确的是()


A. str := ‘abc’ + ‘123’
B. str := "abc" + "123"
C. str := '123' + "abc"
D. fmt.Sprintf("abc%d", 123)
参考答案:BD


6. 【初级】关于协程,下面说法正确是()


A. 协程和线程都可以实现程序的并发执行
B. 线程比协程更轻量级
C. 协程不存在死锁问题
D. 通过channel来进行协程间的通信
参考答案:AD


7. 【中级】关于init函数,下面说法正确的是()


A. 一个包中,可以包含多个init函数
B. 程序编译时,先执行导入包的init函数,再执行本包内的init函数
C. main包中,不能有init函数
D. init函数可以被其他函数调用
参考答案:AB


8. 【初级】关于循环语句,下面说法正确的有()


A. 循环语句既支持for关键字,也支持while和do-while
B. 关键字for的基本使用方法与C/C++中没有任何差异
C. for循环支持continue和break来控制循环,但是它提供了一个更高级的break,可以选择中断哪一个循环
D. for循环不支持以逗号为间隔的多个赋值语句,必须使用平行赋值的方式来初始化多个变量
参考答案:CD

9. 【中级】对于函数定义:
func add(args ...int) int {
sum :=0
for _,arg := range args {
sum += arg
}
return sum
}

下面对add函数调用正确的是()
A. add(1, 2)
B. add(1, 3, 7)
C. add([]int{1, 2})
D. add([]int{1, 3, 7}...)
参考答案:ABD


16. 【初级】关于类型转化,下面语法正确的是()
A.
type MyInt int
var i int = 1
var jMyInt = i
B.
type MyInt int
var i int= 1
var jMyInt = (MyInt)i
C.
type MyInt int
var i int= 1
var jMyInt = MyInt(i)
D.
type MyIntint
var i int= 1
var jMyInt = i.(MyInt)
参考答案:AC


19. 【初级】关于局部变量的初始化,下面正确的使用方式是()


A. var i int = 10
B. var i = 10
C. i := 10
D. i = 10
参考答案:ABC


20. 【初级】关于const常量定义,下面正确的使用方式是()


A.
const Pi float64 = 3.14159265358979323846
const zero= 0.0
B.
const (
size int64= 1024
eof = -1
)
C.
const (
ERR_ELEM_EXIST error = errors.New("element already exists")
ERR_ELEM_NT_EXIST error = errors.New("element not exists")
)



D.
const u, v float32 = 0, 3
const a,b, c = 3, 4, "foo"
参考答案:ABD


22. 【初级】关于布尔变量b的赋值,下面错误的用法是()


A. b = true
B. b = 1
C. b = bool(1)
D. b = (1 == 2)
参考答案:BC

23. 【中级】下面的程序的运行结果是()
func main() {
if (true) {
defer fmt.Printf("1")
} else {
defer fmt.Printf("2")
}
fmt.Printf("3")
}

A. 321
B. 32
C. 31
D. 13
参考答案:C

31. 【初级】关于switch语句,下面说法正确的有()


A. 条件表达式必须为常量或者整数
B. 单个case中,可以出现多个结果选项
C. 需要用break来明确退出一个case
D. 只有在case中明确添加fallthrough关键字,才会继续执行紧跟的下一个case
参考答案:BD






32. 【中级】 golang中没有隐藏的this指针,这句话的含义是()
A. 方法施加的对象显式传递,没有被隐藏起来
B. golang沿袭了传统面向对象编程中的诸多概念,比如继承、虚函数和构造函数
C. golang的面向对象表达更直观,对于面向过程只是换了一种语法形式来表达
D. 方法施加的对象不需要非得是指针,也不用非得叫this
参考答案:ACD

33. 【中级】 golang中的引用类型包括()


A. 数组切片
B. map
C. channel
D. interface
参考答案:ABCD


34. 【中级】 golang中的指针运算包括()


A. 可以对指针进行自增或自减运算
B. 可以通过“&”取指针的地址
C. 可以通过“*”取指针指向的数据
D. 可以对指针进行下标运算
参考答案:BC


35. 【初级】关于main函数(可执行程序的执行起点),下面说法正确的是()


A. main函数不能带参数
B. main函数不能定义返回值
C. main函数所在的包必须为main包
D. main函数中可以使用flag包来获取和解析命令行参数
参考答案:ABCD


36. 【中级】下面赋值正确的是()


A. var x = nil
B. var x interface{} = nil
C. var x string = nil
D. var x error = nil


参考答案:BD





37. 【中级】关于整型切片的初始化,下面正确的是()


A. s := make([]int)
B. s := make([]int, 0)
C. s := make([]int, 5, 10)
D. s := []int{1, 2, 3, 4, 5}


参考答案:BCD


38. 【中级】从切片中删除一个元素,下面的算法实现正确的是()


A.
func (s *Slice)Remove(value interface{})error {
for i, v := range *s {
if isEqual(value, v) {
if i == len(*s)-1 {
*s = (*s)[:i]
} else {
*s = append((*s)[:i], (*s)[i+2:]...)
}
return nil
}
}
return ERR_ELEM_NT_EXIST
}
B.
func (s*Slice)Remove(value interface{}) error {
for i, v:= range *s {
if isEqual(value, v) {
*s =append((*s)[:i],(*s)[i + 1:])
return nil
}
}
returnERR_ELEM_NT_EXIST
}
C.
func (s*Slice)Remove(value interface{}) error {
for i, v:= range *s {
if isEqual(value, v) {
delete(*s, v)
return nil
}
}
returnERR_ELEM_NT_EXIST
}
D.
func (s*Slice)Remove(value interface{}) error {
for i, v:= range *s {
if isEqual(value, v) {
*s =append((*s)[:i],(*s)[i + 1:]...)
return nil
}
}
returnERR_ELEM_NT_EXIST
}
参考答案:D

51. 【初级】对于局部变量整型切片x的赋值,下面定义正确的是()


A.
x := []int{
1, 2, 3,
4, 5, 6,
}
B.
x :=[]int{
1, 2, 3,
4, 5, 6
}
C.
x :=[]int{
1, 2, 3,
4, 5, 6}
D.
x :=[]int{1, 2, 3, 4, 5, 6,}
参考答案:ACD


55. 【初级】关于变量的自增和自减操作,下面语句正确的是()


A.
i := 1
i++
B.
i := 1
j = i++
C.
i := 1
++i
D.
i := 1
i--
参考答案:AD


57. 【中级】关于函数声明,下面语法错误的是()


A. func f(a, b int) (value int, err error)
B. func f(a int, b int) (value int, err error)
C. func f(a, b int) (value int, error)
D. func f(a int, b int) (int, int, error)
参考答案:C

58. 【中级】如果Add函数的调用代码为:
func main() {
var a Integer = 1
var b Integer = 2
var i interface{} = &a
sum := i.(*Integer).Add(b)
fmt.Println(sum)
}

则Add函数定义正确的是()


A.
type Integer int
func (a Integer) Add(b Integer) Integer {
return a + b
}
B.
type Integer int
func (a Integer) Add(b *Integer) Integer {
return a + *b
}
C.
type Integer int
func (a *Integer) Add(b Integer) Integer {
return *a + b
}
D.
typeInteger int
func (a*Integer) Add(b *Integer) Integer {
return *a + *b
}
参考答案:AC

65. 【中级】如果Add函数的调用代码为:
func main() {
var a Integer = 1
var b Integer = 2
var i interface{} = a
sum := i.(Integer).Add(b)
fmt.Println(sum)
}

则Add函数定义正确的是()


A.
type Integer int
func (a Integer) Add(b Integer) Integer {
return a + b
}
B.
typeInteger int
func (aInteger) Add(b *Integer) Integer {
return a + *b
}
C.
typeInteger int
func (a*Integer) Add(b Integer) Integer {
return *a + b
}
D.
typeInteger int
func (a*Integer) Add(b *Integer) Integer {
return *a + *b
}
参考答案:A

72. 【中级】关于GetPodAction定义,下面赋值正确的是()
type Fragment interface {
Exec(transInfo *TransInfo) error
}
type GetPodAction struct {
}
func (g GetPodAction) Exec(transInfo*TransInfo) error {
...
return nil
}

A. var fragment Fragment =new(GetPodAction)


B. var fragment Fragment = GetPodAction


C. var fragment Fragment = &GetPodAction{}


D. var fragment Fragment = GetPodAction{}






参考答案:ACD


81. 【中级】关于GoMock,下面说法正确的是()


A. GoMock可以对interface打桩
B. GoMock可以对类的成员函数打桩
C. GoMock可以对函数打桩
D. GoMock打桩后的依赖注入可以通过GoStub完成


参考答案:AD


82. 【中级】关于接口,下面说法正确的是()


A. 只要两个接口拥有相同的方法列表(次序不同不要紧),那么它们就是等价的,可以相互赋值
B. 如果接口A的方法列表是接口B的方法列表的子集,那么接口B可以赋值给接口A
C. 接口查询是否成功,要在运行期才能够确定
D. 接口赋值是否可行,要在运行期才能够确定


参考答案:ABC


83. 【初级】关于channel,下面语法正确的是()


A. var ch chan int
B. ch := make(chan int)
C. <- ch
D. ch <-
参考答案:ABC


84. 【初级】关于同步锁,下面说法正确的是()


A. 当一个goroutine获得了Mutex后,其他goroutine就只能乖乖的等待,除非该goroutine释放这个Mutex
B. RWMutex在读锁占用的情况下,会阻止写,但不阻止读
C. RWMutex在写锁占用情况下,会阻止任何其他goroutine(无论读和写)进来,整个锁相当于由该goroutine独占
D. Lock()操作需要保证有Unlock()或RUnlock()调用与之对应
参考答案:ABC


85. 【中级】 golang中大多数数据类型都可以转化为有效的JSON文本,下面几种类型除外()


A. 指针
B. channel
C. complex
D. 函数


参考答案:BCD
golang中大多数数据类型都可以转化为有效的JSON文本,除了channel、complex、函数等。
在golang指针中可进行隐式转换,对指针取值,对所指对象进行序列化。
86. 【中级】关于go vendor,下面说法正确的是()
A. 基本思路是将引用的外部包的源代码放在当前工程的vendor目录下面
B. 编译go代码会优先从vendor目录先寻找依赖包
C. 可以指定引用某个特定版本的外部包
D. 有了vendor目录后,打包当前的工程代码到其他机器的$GOPATH/src下都可以通过编译
参考答案:ABD


87. 【初级】 flag是bool型变量,下面if表达式符合编码规范的是()


A. if flag == 1
B. if flag
C. if flag == false
D. if !flag


参考答案:BCD


88. 【初级】 value是整型变量,下面if表达式符合编码规范的是()


A. if value == 0
B. if value
C. if value != 0
D. if !value
参考答案:AC

89. 【中级】关于函数返回值的错误设计,下面说法正确的是()
A. 如果失败原因只有一个,则返回bool
B. 如果失败原因超过一个,则返回error
C. 如果没有失败原因,则不返回bool或error
D. 如果重试几次可以避免失败,则不要立即返回bool或error
参考答案:ABCD

90. 【中级】关于异常设计,下面说法正确的是()
A. 在程序开发阶段,坚持速错,让程序异常崩溃
B. 在程序部署后,应恢复异常避免程序终止
C. 一切皆错误,不用进行异常设计
D. 对于不应该出现的分支,使用异常处理


参考答案:ABD


91. 【中级】关于slice或map操作,下面正确的是()
A.
var s []int
s =append(s,1)
B.
var m map[string]int
m["one"]= 1
C.
var s []int
s =make([]int, 0)
s =append(s,1)
D.
var m map[string]int
m =make(map[string]int)
m["one"]= 1
参考答案:ACD






93. 【中级】关于channel的特性,下面说法正确的是()
A. 给一个 nil channel 发送数据,造成永远阻塞
B. 从一个 nil channel 接收数据,造成永远阻塞
C. 给一个已经关闭的 channel 发送数据,引起 panic
D. 从一个已经关闭的 channel 接收数据,如果缓冲区中为空,则返回一个零值
参考答案:ABCD


94. 【中级】关于无缓冲和有冲突的channel,下面说法正确的是()


A. 无缓冲的channel是默认的缓冲为1的channel
B. 无缓冲的channel和有缓冲的channel都是同步的
C. 无缓冲的channel和有缓冲的channel都是非同步的
D. 无缓冲的channel是同步的,而有缓冲的channel是非同步的
参考答案:D


95. 【中级】关于异常的触发,下面说法正确的是()
A. 空指针解析
B. 下标越界
C. 除数为0
D. 调用panic函数
参考答案:ABCD


96. 【中级】关于cap函数的适用类型,下面说法正确的是()
A. array
B. slice
C. map
D. channel
参考答案:ABD


97. 【中级】关于beego框架,下面说法正确的是()
A. beego是一个golang实现的轻量级HTTP框架
B. beego可以通过注释路由、正则路由等多种方式完成url路由注入
C. 可以使用bee new工具生成空工程,然后使用bee run命令自动热编译
D. beego框架只提供了对url路由的处理,而对于MVC架构中的数据库部分未提供框架支持
参考答案:ABC
98. 【中级】关于goconvey,下面说法正确的是()
A. goconvey是一个支持golang的单元测试框架
B. goconvey能够自动监控文件修改并启动测试,并可以将测试结果实时输出到web界面
C. goconvey提供了丰富的断言简化测试用例的编写
D. goconvey无法与go test集成
参考答案:ABC


99. 【中级】关于go vet,下面说法正确的是()


A. go vet是golang自带工具go tool vet的封装
B. 当执行go vet database时,可以对database所在目录下的所有子文件夹进行递归检测
C. go vet可以使用绝对路径、相对路径或相对GOPATH的路径指定待检测的包
D. go vet可以检测出死代码


参考答案:ACD


100.【中级】关于map,下面说法正确的是()


A. map反序列化时json.unmarshal的入参必须为map的地址
B. 在函数调用中传递map,则子函数中对map元素的增加不会导致父函数中map的修改
C. 在函数调用中传递map,则子函数中对map元素的修改不会导致父函数中map的修改
D. 不能使用内置函数delete删除map的元素
参考答案:A


101. 【中级】关于GoStub,下面说法正确的是()


A. GoStub可以对全局变量打桩
B. GoStub可以对函数打桩
C. GoStub可以对类的成员方法打桩
D. GoStub可以打动态桩,比如对一个函数打桩后,多次调用该函数会有不同的行为


参考答案:ABD


102. 【初级】关于select机制,下面说法正确的是()


A. select机制用来处理异步IO问题
B. select机制最大的一条限制就是每个case语句里必须是一个IO操作
C. golang在语言级别支持select关键字
D. select关键字的用法与switch语句非常类似,后面要带判断条件
参考答案:ABC


103. 【初级】关于内存泄露,下面说法正确的是()


A. golang有自动垃圾回收,不存在内存泄露
B. golang中检测内存泄露主要依靠的是pprof包
C. 内存泄露可以在编译阶段发现
D. 应定期使用浏览器来查看系统的实时内存信息,及时发现内存泄露问题
参考答案:BD

填空题

1. 【初级】声明一个整型变量i__________
参考答案:var i int

2. 【初级】声明一个含有10个元素的整型数组a__________
参考答案:var a [10]int
3. 【初级】声明一个整型数组切片s__________
参考答案:var s []int

4. 【初级】声明一个整型指针变量p__________
参考答案:var p *int

5. 【初级】声明一个key为字符串型value为整型的map变量m__________
参考答案:var m map[string]int


6. 【初级】声明一个入参和返回值均为整型的函数变量f__________
参考答案:var f func(a int) int


7. 【初级】声明一个只用于读取int数据的单向channel变量ch__________
参考答案:var ch <-chan int

8. 【初级】假设源文件的命名为slice.go,则测试文件的命名为__________
参考答案:slice_test.go

9. 【初级】 go test要求测试函数的前缀必须命名为__________
参考答案:Test
10. 【中级】下面的程序的运行结果是__________
for i := 0; i < 5; i++ {
defer fmt.Printf("%d ", i)
}
参考答案:4 3 2 1 0

13. 【中级】下面的程序的运行结果是__________
func main() {
x := 1
{
x := 2
fmt.Print(x)
}
fmt.Println(x)
}
参考答案:21

21. 【中级】下面的程序的运行结果是__________
func main() {
strs := []string{"one","two", "three"}


for _, s := range strs {
go func() {
time.Sleep(1 * time.Second)
fmt.Printf("%s ", s)
}()
}
time.Sleep(3 * time.Second)
}
参考答案:three threethree



32. 【中级】下面的程序的运行结果是__________
func main() {
x := []string{"a", "b", "c"}
for v := range x {
fmt.Print(v)
}
}
参考答案:012
遍历得到的是key


38. 【中级】下面的程序的运行结果是__________
func main() {
x := []string{"a", "b", "c"}
for _, v := range x {
fmt.Print(v)
}
}
参考答案:abc
44. 【初级】下面的程序的运行结果是__________
func main() {
i := 1
j := 2
i, j = j, i
fmt.Printf("%d%d\n", i, j)
}
参考答案:21

50. 【初级】下面的程序的运行结果是__________
func incr(p *int) int {
*p++
return *p
}
func main() {
v := 1
incr(&v)
fmt.Println(v)
}
参考答案:2








59. 【初级】启动一个goroutine的关键字是__________
参考答案:go
60. 【中级】下面的程序的运行结果是__________
type Slice []int
func NewSlice() Slice {
return make(Slice, 0)
}
func (s* Slice) Add(elem int) *Slice {
*s = append(*s, elem)
fmt.Print(elem)
return s
}
func main() {
s := NewSlice()
defer s.Add(1).Add(2).Add(4)
s.Add(3)
}
参考答案:1234



判断题

1. 【初级】数组是一个值类型()
参考答案:True

2. 【初级】使用map不需要引入任何库()
参考答案:True


3. 【中级】内置函数delete可以删除数组切片内的元素()
参考答案:False

4. 【初级】指针是基础类型()
参考答案:False

5. 【初级】 interface{}是可以指向任意对象的Any类型()
参考答案:True
6. 【中级】下面关于文件操作的代码可能触发异常()
file, err := os.Open("test.go")
defer file.Close()
if err != nil {
fmt.Println("open file failed:",err)
return
}
...
参考答案:True


13. 【初级】 Golang不支持自动垃圾回收()
参考答案:False


14. 【初级】 Golang支持反射,反射最常见的使用场景是做对象的序列化()
参考答案:True

15. 【初级】 Golang可以复用C/C++的模块,这个功能叫Cgo()
参考答案:False
16. 【初级】下面代码中两个斜点之间的代码,比如json:"x",作用是X字段在从结构体实例编码到JSON数据格式的时候,使用x作为名字,这可以看作是一种重命名的方式()
type Position struct {
X int `json:"x"`
Y int `json:"y"`
Z int `json:"z"`
}
参考答案:True

21. 【初级】通过成员变量或函数首字母的大小写来决定其作用域()
参考答案:True

22. 【初级】对于常量定义zero(const zero = 0.0),zero是浮点型常量()
参考答案:False True




23. 【初级】对变量x的取反操作是~x()
参考答案:False
24. 【初级】下面的程序的运行结果是xello()
func main() {
str := "hello"
str[0] = 'x'
fmt.Println(str)
}
参考答案:False



str[0]只能get,不能set

29. 【初级】 golang支持goto语句()
参考答案:True
30. 【初级】下面代码中的指针p为野指针,因为返回的栈内存在函数结束时会被释放()
type TimesMatcher struct {
base int
}
func NewTimesMatcher(base int) *TimesMatcher{
return &TimesMatcher{base:base}
}
func main() {
p := NewTimesMatcher(3)
...
}
参考答案:False
GO语言的内存回收机制规定,只要有一个指针指向引用一个变量,那么这个变量就不会被释放,因此在GO语言中返回函数参数或临时变量是安全的。
https://github.com/unknwon/the-way-to-go_ZH_CN/blob/master/eBook/10.2.md




40. 【初级】匿名函数可以直接赋值给一个变量或者直接执行()
参考答案:True


41. 【初级】如果调用方调用了一个具有多返回值的方法,但是却不想关心其中的某个返回值,可以简单地用一个下划线“_”来跳过这个返回值,该下划线对应的变量叫匿名变量()
参考答案:True


42. 【初级】在函数的多返回值中,如果有error或bool类型,则一般放在最后一个()
参考答案:True


43. 【初级】错误是业务过程的一部分,而异常不是()
参考答案:True
链接:https://www.nowcoder.com/questionTerminal/1ed1e4ac2dc142f980ef995b4dd973aa?orderByHotValue=1&page=1&onlyReference=false
来源:牛客网
错误指的是可能出现问题的地方出现了问题,比如打开一个文件时失败,这种情况在人们的意料之中 ;而异常指的是不应该出现问题的地方出现了问题,比如引用了空指针,这种情况在人们的意料之外。可见,错误是业务过程的一部分,而异常不是
Golang中引入error接口类型作为错误处理的标准模式,如果函数要返回错误,则返回值类型列表中肯定包含error。error处理过程类似于C语言中的错误码,可逐层返回,直到被处理。
Golang中引入两个内置函数panic和recover来触发和终止异常处理流程,同时引入关键字defer来延迟执行defer后面的函数。
一直等到包含defer语句的函数执行完毕时,延迟函数(defer后的函数)才会被执行,而不管包含defer语句的函数是通过return的正常结束,还是由于panic导致的异常结束。你可以在一个函数中执行多条defer语句,它们的执行顺序与声明顺序相反。
当程序运行时,如果遇到引用空指针、下标越界或显式调用panic函数等情况,则先触发panic函数的执行,然后调用延迟函数。调用者继续传递panic,因此该过程一直在调用栈中重复发生:函数停止执行,调用延迟执行函数等。如果一路在延迟函数中没有recover函数的调用,则会到达该协程的起点,该协程结束,然后终止其他所有协程,包括主协程(类似于C语言中的主线程,该协程ID为1)。
错误和异常从Golang机制上讲,就是error和panic的区别。很多其他语言也一样,比如C++/Java,没有error但有errno,没有panic但有throw。
Golang错误和异常是可以互相转换的:

  1. 错误转异常,比如程序逻辑上尝试请求某个URL,最多尝试三次,尝试三次的过程中请求失败是错误,尝试完第三次还不成功的话,失败就被提升为异常了。
  2. 异常转错误,比如panic触发的异常被recover恢复后,将返回值中error类型的变量进行赋值,以便上层函数继续走错误处理流程。


44. 【初级】函数执行时,如果由于panic导致了异常,则延迟函数不会执行()
参考答案:False
由panic引发异常以后,程序停止执行,然后调用延迟函数(defer),就像程序正常退出一样。另外recover也是要写在延迟函数中的,如果发生异常延迟函数就不执行了,那就永远无法recover了。

45. 【中级】当程序运行时,如果遇到引用空指针、下标越界或显式调用panic函数等情况,则先触发panic函数的执行,然后调用延迟函数。调用者继续传递panic,因此该过程一直在调用栈中重复发生:函数停止执行,调用延迟执行函数。如果一路在延迟函数中没有recover函数的调用,则会到达该携程的起点,该携程结束,然后终止其他所有携程,其他携程的终止过程也是重复发生:函数停止执行,调用延迟执行函数()
参考答案:False
链接:https://www.nowcoder.com/questionTerminal/e400d457afb14916a4a791fc93a3669c
来源:牛客网


当内置的panic()函数调用时,外围函数或方法的执行会立即终止。然后,任何延迟执行(defer)的函数或方法都会被调用,就像其外围函数正常返回一样。最后,调用返回到该外围函数的调用者,就像该外围调用函数或方法调用了panic()一样,因此该过程一直在调用栈中重复发生:函数停止执行,调用延迟执行函数等。当到达main()函数时不再有可以返回的调用者,因此这个过程会终止,并将包含传入原始panic()函数中的值的调用栈信息输出到os.Stderr。

46. 【初级】同级文件的包名不允许有多个()
参考答案:True

47. 【中级】可以给任意类型添加相应的方法()
参考答案:False
在 Go 语言中,结构体就像是类的一种简化形式,那么面向对象程序员可能会问:类的方法在哪里呢?在 Go 中有一个概念,它和方法有着同样的名字,并且大体上意思相同:Go 方法是作用在接收者(receiver)上的一个函数,接收者是某种类型的变量。因此方法是一种特殊类型的函数。
接收者类型可以是(几乎)任何类型,不仅仅是结构体类型:任何类型都可以有方法,甚至可以是函数类型,可以是 int、bool、string 或数组的别名类型。但是接收者不能是一个接口类型(参考 第 11 章),因为接口是一个抽象定义,但是方法却是具体实现;如果这样做会引发一个编译错误:invalid receiver type…
最后接收者不能是一个指针类型,但是它可以是任何其他允许类型的指针。
https://github.com/unknwon/the-way-to-go_ZH_CN/blob/master/eBook/10.6.md


类型和作用在它上面定义的方法必须在同一个包里定义,这就是为什么不能在 int、float 或类似这些的类型上定义方法。如果直接给int添加method会报错
`cannot define new methods on non-local type int`


链接:https://www.nowcoder.com/questionTerminal/c894f2df2ef349c6bb92ea6efafbf3a3
来源:牛客网
如果`type Interger int`定义别名就可以了,也就是需要是自定义的类型才能定义方法

48. 【初级】 golang虽然没有显式的提供继承语法,但是通过匿名组合实现了继承()
参考答案:True

49. 【初级】使用for range迭代map时每次迭代的顺序可能不一样,因为map的迭代是随机的()
参考答案:True


50. 【初级】 switch后面可以不跟表达式()
参考答案:True
switch 语句的第二种形式是不提供任何被判断的值(实际上默认为判断是否为 true),然后在每个 case 分支中进行测试不同的条件。当任一分支的测试结果为 true 时,该分支的代码会被执行。这看起来非常像链式的 if-else 语句,但是在测试条件非常多的情况下,提供了可读性更好的书写方式。
switch {
case condition1:
...
case condition2:
...
default:
...
}

51. 【中级】结构体在序列化时非导出变量(以小写字母开头的变量名)不会被encode,因此在decode时这些非导出变量的值为其类型的零值()
参考答案:True
序列化通常将类型结构传入标准库或第三方包,类型结构中没有大写的变量未导出,对第三方包不可见,无法进行任何操作,依旧是默认的零值。

52. 【初级】 golang中没有构造函数的概念,对象的创建通常交由一个全局的创建函数来完成,以NewXXX来命名()
参考答案:True
Go 语言不支持面向对象编程语言中那样的构造子方法,但是可以很容易的在 Go 中实现 “构造子工厂”方法。为了方便通常会为类型定义一个工厂,按惯例,工厂的名字以 new 或 New 开头。假设定义了如下的 File 结构体类型:
type File struct {
fd int // 文件描述符
name string // 文件名
}
下面是这个结构体类型对应的工厂方法,它返回一个指向结构体实例的指针:
func NewFile(fd int, name string) *File {
if fd < 0 {
return nil
}


return &File{fd, name}
}
然后这样调用它:
f := NewFile(10, "./test.txt")
53. 【中级】当函数deferDemo返回失败时,并不能destroy已create成功的资源()
func deferDemo() error {
err := createResource1()
if err != nil {
return ERR_CREATE_RESOURCE1_FAILED
}
defer func() {
if err != nil {
destroyResource1()
}
}()


err = createResource2()
if err != nil {
return ERR_CREATE_RESOURCE2_FAILED
}
defer func() {
if err != nil {
destroyResource2()
}
}()


err = createResource3()
if err != nil {
return ERR_CREATE_RESOURCE3_FAILED
}
return nil
}
参考答案:False
链接:https://www.nowcoder.com/questionTerminal/6501a1f6a0804ab59228cb05b09fce84
来源:牛客网


答案是能成功释放创建成功的资源 题目说的是释放〈已成功〉创建的资源,这里的err是引用变量,放在defer语句里err引用的是同一个地址,前面资源创建成功时err是nil,后面只要任意一次创建失败这个err值就不为nil(有点类似闭包的问题),所以前面的defer语句判断成立会释放资源

80. 【中级】 channel本身必然是同时支持读写的,所以不存在单向channel()
参考答案:False
链接:https://www.nowcoder.com/questionTerminal/afaa9c496463416693e111f9d259c924
来源:牛客网


有单向通道。一般作为函数的参数或者返回值。 单向通道都是由双向通道转换而来,不能自己声明单向通道,没有意义。 单向通道作为函数或者方法的参数时,表示该函数或者方法作用域内,只能执行接收或者发送操作。 单向通道作为函数或者方法返回值时,表示调用者得到该通道后只能做单向操作。

81. 【初级】 import后面的最后一个元素是包名()
参考答案:False
链接:https://www.nowcoder.com/questionTerminal/f7c0e0d2c68a4a95b62576ec63934f3d
来源:牛客网


  • import后面跟的是包的路径,而不是包名;
  • 同一个目录下可以有多个.go文件,但是只能有一个包;
  • 使用第三方库时,先将源码编译成.a文件放到临时目录下,然后去链接这个.a文件,而不是go install安装的那个.a文件;
  • 使用标准库时,直接链接.a文件,即使修改了源码,也不会从新编译源码;
  • 不管使用的是标准库还是第三方库,源码都是必须存在的,即使使用的是.a文件。

最近在很多地方看到了golang的面试题,看到了很多人对Golang的面试题心存恐惧,也是为了复习基础,我把解题的过程总结下来。
面试题
1 写出下面代码输出内容。

考点:defer执行顺序


解答:


defer 是后进先出。
panic 需要等defer 结束后才会向上传递。 出现panic恐慌时候,会先按照defer的后入先出的顺序执行,最后才会执行panic。

打印后
打印中
打印前
panic: 触发异常
2 以下代码有什么问题,说明原因。

package main


import "fmt"


type student struct {
Name string
Age int
}
func main(){
pase_student()
}


func pase_student() {
m := make(map[string]*student)
stus := []student{
{Name: "zhou", Age: 24},
{Name: "li", Age: 23},
{Name: "wang", Age: 22},
}
//错误写法 stu.Age 赋值的是副本,不影响原有变量
for _, stu := range stus {
stu.Age = stu.Age + 10
}
fmt.Println(stus)
//错误写法 与Java的foreach一样,都是使用副本的方式。
// 所以m[stu.Name]=&stu实际上一致指向同一个指针,
// 最终该指针的值为遍历的最后一个struct的值拷贝。
// 就像想修改切片元素的属性:
for _, stu := range stus {
m[stu.Name] = &stu
}
for k, v := range m {
fmt.Println(k, "=>", v.Name)
}
fmt.Println(m)
//正确写法
for i := 0; i < len(stus); i++ {
m[stus[i].Name] = &stus[i]
}
for k, v := range m {
fmt.Println(k, "=>", v.Name)
}
}


考点:foreach


解答:



3 下面的代码会输出什么,并说明原因

package main


import (
"fmt"
"runtime"
"sync"
)


func main() {
runtime.GOMAXPROCS(1)
wg := sync.WaitGroup{}
wg.Add(20)


for i := 0; i < 10; i++ {
go func() {
fmt.Println("A: ", i)
wg.Done()
}()
}


for i := 0; i < 10; i++ {
go func(i int) {
fmt.Println("B: ", i)
wg.Done()
}(i)
}
wg.Wait()
}
考点:go执行的随机性和闭包


解答:


谁也不知道执行后打印的顺序是什么样的,所以只能说是随机数字。 但是A:均为输出10,B:从0~9输出(顺序不定)。 第一个go func中i是外部for的一个变量,地址不变化。遍历完成后,最终i=10。 故go func执行时,i的值始终是10。

第二个go func中i是函数参数,与外部for中的i完全是两个变量。 尾部(i)将发生值拷贝,go func内部指向值拷贝地址。



4 下面代码会输出什么?

package main


import (
"fmt"
)


type People struct{}
func (p *People) ShowA() {
fmt.Println("showA")
p.ShowB()
}


func (p*People) ShowB() {
fmt.Println("showB")
}


type Teacher struct {
People
}


func (t*Teacher) ShowB() {
fmt.Println("teacher showB")
}


func main() {
t := Teacher{}
t.ShowA()
}


考点:go的组合继承


解答:


这是Golang的组合模式,可以实现OOP的继承。 被组合的类型People所包含的方法虽然升级成了外部类型Teacher这个组合类型的方法(一定要是匿名字段),但它们的方法(ShowA())调用时接受者并没有发生变化。 此时People类型并不知道自己会被什么类型组合,当然也就无法调用方法时去使用未知的组合者Teacher类型的功能。



5 下面代码会触发异常吗?请详细说明

package main


import (
"fmt"
"runtime"
)
func main() {
runtime.GOMAXPROCS(1)
int_chan := make(chan int, 1)
string_chan := make(chan string, 1)
int_chan <- 1
string_chan <- "hello"
select {
case value := <-int_chan:
fmt.Println(value)
case value := <-string_chan:
panic(value)
}
}

考点:select随机性


解答:


select会随机选择一个可用通用做收发操作。 所以代码是有可能触发异常,也有可能不会。 单个chan如果无缓冲时,将会阻塞。但结合 select可以在多个chan间等待执行。有三点原则:

select 中只要有一个case能return,则立刻执行。
当如果同一时间有多个case均能return则伪随机方式抽取任意一个执行。
如果没有一个case能return则可以执行”default”块。



6 下面代码输出什么?

package main


import (
"fmt"
)
func calc(index string, a, b int) int {
ret := a + b
fmt.Println(index, a, b, ret)
return ret
}


func main() {
a := 1
b := 2
defer calc("1", a, calc("10", a, b))
a = 0
defer calc("2", a, calc("20", a, b))
b = 1
}

考点:defer执行顺序


解答:



这道题类似第1题 需要注意到defer执行顺序和值传递 index:1肯定是最后执行的,但是index:1的第三个参数是一个函数,所以最先被调用


calc("10",1,2)==>10,1,2,3 执行index:2时,与之前一样,需要先调用calc("20",0,2)==>20,0,2,2 执行到b=1时候开始调用,index:2==>calc("2",0,2)==>2,0,2,2最后执行index:1==>calc("1",1,3)==>1,1,3,4
(base) ruizhidu@stone awesomeProject3 % go run main.go
10 1 2 3
20 0 2 2
2 0 2 2
1 1 3 4
7 请写出以下输入内容

package main


import (
"fmt"
)


func main() {
s := make([]int, 0)
s = append(s, 1, 2, 3)
fmt.Println(s)


s2 := make([]int, 5)
s2 = append(s2, 1, 2, 3)
fmt.Println(s2)
}

考点:make默认值和append


解答:
make初始化是由默认值的哦,此处默认值为0



8 下面的代码有什么问题?

package main


import (
//"fmt"
"fmt"
"sync"
)


func main() {
user:=UserAges{ages: map[string]int{}}
user.Add("liming",10)
age:=user.Get("liming")
fmt.Println(age)


age2:=user.Get2("liming")
fmt.Println(age2)
}
type UserAges struct {
ages map[string]int
sync.Mutex
}


func(ua*UserAges)Add(name string, age int) {
ua.Lock()
defer ua.Unlock()
ua.ages[name] = age
}
// 可能会报错,也不一定。fatal error: concurrent map read and map write.
func (ua *UserAges) Get(name string) int {
if age, ok := ua.ages[name]; ok {
return age
}
return -1
}
//优化后
func (ua *UserAges)Get2(name string)int {
ua.Lock()
defer ua.Unlock()


if age, ok := ua.ages[name]; ok {
return age
}
return -1
}



考点:map线程安全
解答:
可能会报错,也不一定。fatal error: concurrent map read and map write.
9. 下面的迭代会有什么问题?

func (set *threadSafeSet)Iter()<-chan interface{} {
ch := make(chan interface{})
go func() {
set.RLock()
for elem := range set.s {
ch <- elem
}
close(ch)
set.RUnlock()
}()
return ch
}



考点:chan缓存池


解答:


看到这道题,我也在猜想出题者的意图在哪里。 chan?sync.RWMutex?go?chan缓存池?迭代? 所以只能再读一次题目,就从迭代入手看看。 既然是迭代就会要求set.s全部可以遍历一次。但是chan是为缓存的,那就代表这写入一次就会阻塞。 我们把代码恢复为可以运行的方式,看看效果

package main


import (
"fmt"
"sync"
)


//下面的迭代会有什么问题?
type threadSafeSet struct {
sync.RWMutex
s []interface{}
}


func (set *threadSafeSet)Iter() <-chan interface{} {
ch := make(chan interface{}) // 解除注释看看!
//ch := make(chan interface{}, len(set.s))
go func() {
set.RLock()
for elem, value := range set.s {
ch <- elem
println("Iter:", elem, value)
}
close(ch)
set.RUnlock()
}()
return ch
}


func main() {
th := threadSafeSet{
s: []interface{}{"1", "2"},
}
v := <-th.Iter()
fmt.Sprintf("%s%v", "ch", v)
}


1.当 ch := make(chan interface{}) 时 ,运行结果为:
(base) ruizhidu@stone awesomeProject3 % go run main.go
Iter: 0 (0x10acca0,0x10eb690)


2.当 ch := make(chan interface{}, len(set.s)) 时 ,运行结果为:
(base) ruizhidu@stone awesomeProject3 % go run main.go
Iter: 0 (0x10acca0,0x10eb690)
Iter: 1 (0x10acca0,0x10eb6a0)


10 以下代码能编译过去吗?为什么?



package main


import "fmt"


type People interface {
Speak(string) string
}


type Stduent struct{}


func(stu *Stduent)Speak(think string)(talk string) {
if think == "bitch" {
talk = "You are a good boy"
} else {
talk = "hi"
}
return
}


func main() {
//错误
var peo People = Stduent{}
//改为指针或者方法改为func(stu Stduent)Speak(think string)
//var peo People = &Stduent{}
think := "bitch"
fmt.Println(peo.Speak(think))
}

考点:golang的方法集
解答:
编译不通过! 做错了!?说明你对golang的方法集还有一些疑问。 一句话:golang的方法集仅仅影响接口实现和方法表达式转化,与通过实例或者指针调用方法无关。

11 以下代码打印出来什么内容,说出为什么。

package main


import "fmt"


type People interface {
Show()
}


type Student struct{}


func(stu *Student)Show() {
}


func live() People {
var stu *Student
return stu
}


func main() {
if live() == nil {
fmt.Println("AAAAAAA")
} else {
fmt.Println("BBBBBBB")
}
}





考点:interface内部结构


解答:


很经典的题! 这个考点是很多人忽略的interface内部结构。 go中的接口分为两种一种是空的接口类似这样:
var in interface{}
另一种如题目:

type People interface {


Show()


}
他们的底层结构如下:

type eface struct { //空接口


_type *_type //类型信息


data unsafe.Pointer //指向数据的指针(go语言中特殊的指针类型unsafe.Pointer类似于c语言中的void*)}


typeiface struct { //带有方法的接口


tab *itab //存储type信息还有结构实现方法的集合


data unsafe.Pointer //指向数据的指针(go语言中特殊的指针类型unsafe.Pointer类似于c语言中的void*)}


type_type struct {


size uintptr //类型大小


ptrdata uintptr //前缀持有所有指针的内存大小


hash uint32 //数据hash值


tflag tflag


align uint8 //对齐


fieldalign uint8 //嵌入结构体时的对齐


kind uint8 //kind 有些枚举值kind等于0是无效的


alg *typeAlg //函数指针数组,类型实现的所有方法


gcdata *byte str nameOff


ptrToThis typeOff


}type itab struct {


inter *interfacetype //接口类型


_type *_type //结构类型


link *itab


bad int32


inhash int32


fun [1]uintptr //可变大小方法集合}
可以看出iface比eface 中间多了一层itab结构。 itab 存储_type信息和[]fun方法集,从上面的结构我们就可得出,因为data指向了nil 并不代表interface 是nil, 所以返回值并不为空,这里的fun(方法集)定义了接口的接收规则,在编译的过程中需要验证是否实现接口 结果:

BBBBBBB


12.是否可以编译通过?如果通过,输出什么?
package main


func main() {


i := GetValue()
switch i.(type) {
case int:
println("int")
case string:
println("string")
case interface{}:
println("interface")
default:
println("unknown")
}
}


func GetValue()int {
return 1
}

解析
考点:type
编译失败,因为type只能使用在interface



13.下面函数有什么问题?

func funcMui(x,y int)(sum int,error){
return x+y,nil
}

解析
考点:函数返回值命名
在函数有多个返回值时,只要有一个返回值有指定命名,其他的也必须有命名。 如果返回值有有多个返回值必须加上括号; 如果只有一个返回值并且有命名也需要加上括号; 此处函数第一个返回值有sum名称,第二个未命名,所以错误。
14.是否可以编译通过?如果通过,输出什么?

package main

func main() {
//执行顺序
//1
println(DeferFunc1(1))
//2
println(DeferFunc2(1))
//3
println(DeferFunc3(1))
}


func DeferFunc1(i int)(t int) {
t = i
defer func() {
t += 3
}()
return t
}


func DeferFunc2(i int)int {
t := i
defer func() {
t += 3
}()
return t
}


func DeferFunc3(i int)(t int) {
defer func() {
t += i
}()
return 2
}
执行结果:
(base) ruizhidu@stone awesomeProject3 % go run main.go
4
1
3

解析



考点:defer和函数返回值


需要明确一点是defer需要在函数结束前执行。 函数返回值名字会在函数起始处被初始化为对应类型的零值并且作用域为整个函数 DeferFunc1有函数返回值t作用域为整个函数,在return之前defer会被执行,所以t会被修改,返回4; DeferFunc2函数中t的作用域为函数,返回1;DeferFunc3返回3
15.是否可以编译通过?如果通过,输出什么?

package main


import "fmt"


func main() {
//错误
//list := new([]int)
//正确
list := make([]int,0)
list = append(list, 1)
fmt.Println(list)
}

解析


考点:new
list := make([]int,0)
16.是否可以编译通过?如果通过,输出什么?

package main
import "fmt"
func main() {
s1 := []int{1, 2, 3}
s2 := []int{4, 5}
//错误
//s1 = append(s1, s2)
//正确
s1 = append(s1, s2...)
fmt.Println(s1)
}


解析


考点:append
append切片时候别漏了'…'
17.是否可以编译通过?如果通过,输出什么?

package main
import (
"fmt"
"reflect"
)
func main() {
sn1 := struct {
age int
name string
}{age: 11, name: "qq"}
sn2 := struct {
age int
name string
}{age: 11, name: "qq"}
if sn1 == sn2 {
fmt.Println("sn1== sn2")
}


sm1 := struct {
age int
m map[string]string
}{age: 11, m: map[string]string{"a": "1"}}


sm2 := struct {
age int
m map[string]string
}{age: 11, m: map[string]string{"a": "1"}}
// 错误 sm1 里面map 是引用类型,不能直接比较
//if sm1 == sm2 {
//正确 sm1 里面map 是引用类型,改为
if reflect.DeepEqual(sm1, sm2) {
fmt.Println("sm1== sm2")
}
}



解析


考点:结构体比较


进行结构体比较时候,只有相同类型的结构体才可以比较,结构体是否相同不但与属性类型个数有关,还与属性顺序相关。

sn3:= struct {


name string


age int


}


{age:11,name:"qq"}
sn3与sn1就不是相同的结构体了,不能比较。 还有一点需要注意的是结构体是相同的,但是结构体属性中有不可以比较的类型,如map,slice。 如果该结构属性都是可以比较的,那么就可以使用“==”进行比较操作。
可以使用reflect.DeepEqual进行比较

if reflect.DeepEqual(sn1, sm) {


fmt.Println("sn1==sm")


}else {


fmt.Println("sn1!=sm")


}
所以编译不通过: invalid operation: sm1 == sm2
18.是否可以编译通过?如果通过,输出什么?

package main


import "fmt"
func Foo(x interface{}) {
if x == nil {
fmt.Println("empty interface")
return
}
fmt.Println("non-empty interface")
}


func main() {
var x *int = nil
Foo(x)
}

执行结果
(base) ruizhidu@stone awesomeProject3 % go run main.go
non-empty interface
解析
考点:interface内部结构
19.是否可以编译通过?如果通过,输出什么?

package main


import "fmt"


func GetValue(m map[int]string, id int)(string, bool) {
if _, exist := m[id]; exist {
return "存在数据", true
}
return nil, false
}
func main() {
intmap := map[int]string{
1: "a",
2: "bb",
3: "ccc",
}
v, err := GetValue(intmap, 3)
fmt.Println(v, err)
}



解析


考点:函数返回值类型
nil 可以用作 interface、function、pointer、map、slice 和 channel 的“空值”。但是如果不特别指定的话,Go 语言不能识别类型,所以会报错。报:cannot use nil as type string in return argument.
20.是否可以编译通过?如果通过,输出什么?

package main


import "fmt"


const (
x = iota
y
z = "zz"
k
p = iota)


func main() {
fmt.Println(x, y, z, k, p)
}
(base) ruizhidu@stone awesomeProject3 % go run main.go
0 1 zz zz 4







解析
考点:iota

21.编译执行下面代码会出现什么?



package main
var(
//错误
//size :=1024
//正确
size =1024
max_size = size*2
)


func main() {
println(size, max_size)
}



解析


考点:变量简短模式


变量简短模式限制:

定义变量同时显式初始化


不能提供数据类型


只能在函数内部使用


结果:
syntax error: unexpected :=
22.下面函数有什么问题?

package main


const cl = 100
var bl = 123


func main() {
println(&bl, bl)
//错误 常量不同于变量的在运行期分配内存,常量通常会被编译器在预处理阶段直接展开,作为指令数据使用,
println(&cl, cl)
}




解析


考点:常量


常量不同于变量的在运行期分配内存,常量通常会被编译器在预处理阶段直接展开,作为指令数据使用,
cannot take the address of cl
23.编译执行下面代码会出现什么?

package main


func main() {
for i := 0; i < 10; i++ {
loop:
println(i)
}
// 错误 goto不能跳转到其他函数或者内层代码
goto loop


}







解析


考点:goto


goto不能跳转到其他函数或者内层代码
goto loop jumps intoblock starting at
24.编译执行下面代码会出现什么?

package main


import "fmt"


func main() {
type MyInt1 int
type MyInt2 = int
var i int = 9
//错误
//var i1 MyInt1 = i
//正确
var i1 MyInt1 = MyInt1(i)
var i2 MyInt2 = i
fmt.Println(i1, i2)
}



解析


考点:**Go 1.9 新特性 Type Alias **


基于一个类型创建一个新类型,称之为defintion;基于一个类型创建一个别名,称之为alias。 MyInt1为称之为defintion,虽然底层类型为int类型,但是不能直接赋值,需要强转; MyInt2称之为alias,可以直接赋值。
结果:
cannot use i (typeint) astype MyInt1 in assignment
25.编译执行下面代码会出现什么?

package main


import "fmt"


type User struct {
}
type MyUser1 User
type MyUser2 = User
func(i MyUser1)m1() {
fmt.Println("MyUser1.m1")
}


func(i User)m1(){
fmt.Println("User.m1")
}
func(i User)m2(){
fmt.Println("User.m2")
}


func main() {
var i1 MyUser1
var i2 MyUser2
i1.m1()
i2.m2()
}


(base) ruizhidu@stone awesomeProject3 % go run main.go
MyUser1.m1
User.m2







解析


考点:**Go 1.9 新特性 Type Alias **


因为MyUser2完全等价于User,所以具有其所有的方法,并且其中一个新增了方法,另外一个也会有。 但是
i1.m2()
是不能执行的,因为MyUser1没有定义该方法。 结果:
MyUser1.m1User.m2
26.编译执行下面代码会出现什么?

package main


import"fmt"


type T1 struct {
}


func(t T1)m1() {
fmt.Println("T1.m1")
}


type T2= T1


type MyStruct struct {
T1
T2
}


func main() {
my := MyStruct{}
my.m1()
}





修改后:


package main


import"fmt"


type T1 struct {
}


func(t T1)m1() {
fmt.Println("T1.m1")
}


type T2= T1


type MyStruct struct {
T1
T2
}


func main() {
my := MyStruct{}
//错误
//my.m1()
//改为
my.T1.m1()
my.T2.m1()
}



解析


考点:**Go 1.9 新特性 Type Alias **


是不能正常编译的,异常:
ambiguousselectormy.m1
结果不限于方法,字段也也一样;也不限于type alias,type defintion也是一样的,只要有重复的方法、字段,就会有这种提示,因为不知道该选择哪个。 改为:

my.T1.m1()


my.T2.m1()
type alias的定义,本质上是一样的类型,只是起了一个别名,源类型怎么用,别名类型也怎么用,保留源类型的所有方法、字段等。
27.编译执行下面代码会出现什么?

package main


import (
"errors"
"fmt")


var ErrDidNotWork = errors.New("did not work")


//修改前 错误 main 函数执行fmt.Println(DoTheThing(true)),结果为nil
//func DoTheThing(reallyDoIt bool)(err error) {
// if reallyDoIt {
// //这里的err := 是if里面的局部变量
// result, err := tryTheThing()
// if err != nil || result != "it worked" {
// err = ErrDidNotWork
// }
// }
// //返回值声明了err 为nil
// return err
//}


//改正后 main 函数执行fmt.Println(DoTheThing(true)),结果为did not work
func DoTheThing(reallyDoIt bool)(err error) {
var result string
if reallyDoIt {
result, err = tryTheThing()
if err != nil || result != "it worked" {
err = ErrDidNotWork
}
}
return err
}


func tryTheThing()(string,error) {
return "", ErrDidNotWork
}


func main() {
fmt.Println(DoTheThing(true))
fmt.Println(DoTheThing(false))
}
修改前 执行结果为:
(base) ruizhidu@stone awesomeProject3 % go run main.go
<nil>
<nil>
修改后执行结果为:
(base) ruizhidu@stone awesomeProject3 % go run main.go
did not work
<nil>




解析


考点:变量作用域


因为 if 语句块内的 err 变量会遮罩函数作用域内的 err 变量,结果:
改为:

func DoTheThing(reallyDoIt bool)(errerror) {


varresult string


ifreallyDoIt {


result, err =tryTheThing()


if err!= nil || result != "it worked" {


err = ErrDidNotWork


}


} return err


}
28.编译执行下面代码会出现什么?

package main
//修改前
//func test() []func() {
// var funs []func()
// for i := 0; i < 2; i++ {
// funs = append(funs, func() {
// println(&i, i)
// })
// }
// return funs
//}


//修改后
func test() []func() {
var funs []func()
for i := 0; i < 2; i++ {
x := i
funs = append(funs, func() {
println(&x, x)
})
}
return funs
}
func main() {
funs := test()
for _, f :=range funs{
f()
}
}
//修改前
(base) ruizhidu@stone awesomeProject3 % go run main.go
0xc000072000 2
0xc000072000 2
//修改后
(base) ruizhidu@stone awesomeProject3 % go run main.go
0xc000014098 0
0xc0000140b0 1





解析


考点:闭包延迟求值


for循环复用局部变量i,每一次放入匿名函数的应用都是想一个变量。 结果:

0xc042046000 2


0xc042046000 2
如果想不一样可以改为:



29.编译执行下面代码会出现什么?

package main


func test(x int)(func(),func()) {
return func() {
println(x)
x += 10
}, func() {
println(x)
}
}


func main() {
a, b := test(100)
a()
b()
}

(base) ruizhidu@stone awesomeProject3 % go run main.go
100
110




解析


考点:闭包引用相同变量*


结果:

100


110
30. 编译执行下面代码会出现什么?


package main


import (
"fmt"
"reflect"


//"reflect"
)


//func main() {
// defer func() {
// if err := recover(); err != nil {
// fmt.Println(err)
// } else {
// fmt.Println("fatal")
// }
// }()
// defer func() {
// panic("defer panic")
// }()
// panic("panic")
//}


func main() {
defer func() {
if err := recover(); err != nil {
fmt.Println("++++")
f := err.(func() string)
fmt.Println(err, f(), reflect.TypeOf(err).Kind().String())
} else {
fmt.Println("fatal")
}
}()
defer func() {
panic(func() string {
return "defer panic"
})
}()
panic("panic")
}








解析


考点:panic仅有最后一个可以被revover捕获
触发panic("panic")后顺序执行defer,但是defer中还有一个panic,所以覆盖了之前的panic("panic")

网址:http://www.qukuailianxueyuan.io/