1.下面的代码输出什么?
type N int
func (n *N) test(){
fmt.Println(*n)
}
func main() {
var n N = 10
p := &n
n++
f1 := n.test
n++
f2 := p.test
n++
fmt.Println(n)
f1()
f2()
}
参考答案及解析:13 13 13。知识点:方法值。当目标方法的接收者是指针类型时,那么被复制的就是指针。
2.下面哪一行代码会 panic,请说明原因?
package main
func main() {
var m map[int]bool // nil
_ = m[123]
var p *[5]string // nil
for range p {
_ = len(p)
}
var s []int // nil
_ = s[:]
s, s[0] = []int{1, 2}, 9
}
参考答案及解析:第 12 行。因为左侧的 s[0] 中的 s 为 nil。
3.下面哪一行代码会 panic,请说明原因?
package main
type T struct{}
func (*T) foo() {
}
func (T) bar() {
}
type S struct {
*T
}
func main() {
s := S{}
_ = s.foo
s.foo()
_ = s.bar
}
参考答案及解析:第 19 行,因为 s.bar 将被展开为 (*s.T).bar,而 s.T 是个空指针,解引用会 panic。
可以使用下面代码输出 s:
func main() {
s := S{}
fmt.Printf("%#v",s) // 输出:main.S{T:(*main.T)(nil)}
}
4.下面的代码有什么问题?
type data struct {
sync.Mutex
}
func (d data) test(s string) {
d.Lock()
defer d.Unlock()
for i:=0;i<5 ;i++ {
fmt.Println(s,i)
time.Sleep(time.Second)
}
}
func main() {
var wg sync.WaitGroup
wg.Add(2)
var d data
go func() {
defer wg.Done()
d.test("read")
}()
go func() {
defer wg.Done()
d.test("write")
}()
wg.Wait()
}
参考答案及解析:锁失效。将 Mutex 作为匿名字段时,相关的方法必须使用指针接收者,否则会导致锁机制失效。
修复代码:
func (d *data) test(s string) { // 指针接收者
d.Lock()
defer d.Unlock()
for i:=0;i<5 ;i++ {
fmt.Println(s,i)
time.Sleep(time.Second)
}
}
或者可以通过嵌入 *Mutex 来避免复制的问题,但需要初始化。
type data struct {
*sync.Mutex // *Mutex
}
func (d data) test(s string) { // 值方法
d.Lock()
defer d.Unlock()
for i := 0; i < 5; i++ {
fmt.Println(s, i)
time.Sleep(time.Second)
}
}
func main() {
var wg sync.WaitGroup
wg.Add(2)
d := data{new(sync.Mutex)} // 初始化
go func() {
defer wg.Done()
d.test("read")
}()
go func() {
defer wg.Done()
d.test("write")
}()
wg.Wait()
}