开启掘金成长之旅!这是我参与「掘金日新计划 · 12 月更文挑战」的第24天,点击查看活动详情
今天来学习下Go常见的习题问题,也是面试中可能会遇到的,让我们来一起学习吧~
习题
defer的输出顺序
观察下面代码,写出代码输出的内容,可以先自己想下,再看后面的答案
package main
import (
"fmt"
)
func main() {
defer_call()
}
func defer_call() {
defer func() { fmt.Println("打印前") }()
defer func() { fmt.Println("打印中") }()
defer func() { fmt.Println("打印后") }()
panic("触发异常")
}
答案如下
打印后
打印中
打印前
panic: 触发异常
这里考了defer的执行顺序,是类似栈的顺序,先进后出,当出现panic方法的时候程序就会终止运行,就会按照defer的先进后出顺序执行,所以打印的是后、中、前,最后才会执行panic方法
for range 的错误
观察下面代码,写出代码输出的内容,可以先自己想下,再看后面的答案
func main() {
slice := []int{0,1,2,3}
m := make(map[int]*int)
for key,val := range slice {
m[key] = &val
}
for k,v := range m {
fmt.Println(k,"->",*v)
}
}
答案如下:
0 -> 3
1 -> 3
2 -> 3
3 -> 3
这里为什么没有输出0 1 2 3呢?这是因为for range循环的时候会创建每个元素的副本,而不是元素的引用,所以 m[key] = &val取的都是val的地址,最后val被赋值为3,所以map中的所有元素都指向val变量的地址,也就是输出的值都是val最后赋值的值为3
正确的写法: 通过给变量val赋值到新的变量value中,就可以把每次的值给保存下来
func main() {
slice := []int{0,1,2,3}
m := make(map[int]*int)
for key,val := range slice {
value := val
m[key] = &value
}
for k,v := range m {
fmt.Println(k,"===>",*v)
}
}
0 -> 0
1 -> 1
2 -> 2
3 -> 3
slice 输出
观察下面两个代码,写出代码输出的内容,可以先自己想下,再看后面的答案
// 1.
func main() {
s := make([]int, 5)
s = append(s, 1, 2, 3)
fmt.Println(s)
}
// 2.
func main() {
s := make([]int,0)
s = append(s,1,2,3,4)
fmt.Println(s)
}
输出结果:
[0 0 0 0 0 1 2 3]
[1 2 3 4]
第一个程序中的切片一开始初始化有5个位置,初值为0,后面通过append添加元素会在后面进行添加
下列代码有啥问题?
func test(x,y int)(sum int,error){
return x+y,nil
}
test函数的第二个参数返回值没有命名,在函数有多个返回值的时候,只要有个一个返回值有命名,其他的也必须要有,并且多个返回值需要加上括号,如果只有一个返回值并且命名也是需要加上括号,只有没有命令单个返回值才可以不加括号
new()与make()区别
new(T) 和 make(T,args) 是 Go 语言内建函数,用来分配内存,但适用的类型不同。
new(T) 会为 T 类型的新值分配已置零的内存空间,并返回地址(指针),即类型为 *T的值。换句话说就是,返回一个指针,该指针指向新分配的、类型为 T 的零值,适用于值类型,如数组、结构体等。
make(T,args) 返回初始化之后的 T 类型的值,这个值并不是 T 类型的零值,也不是指针 *T,是经过初始化之后的 T 的引用。make() 只适用于 slice、map 和 channel.
总结
今天浅谈了Go的习题(一),主要介绍了GO面试中会出现的问题,接下来会继续分享其他的习题的相关知识,对于一个刚入门的我来说,还有许多地方需要学习,有错误的地方欢迎大家指出,共同进步!!