Go面试题60
1: 下面哪一行代码会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
}
2: 下面的代码有什么问题?
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()
}
参考答案及解析
1: 答案及解析: 第19行, 因为s.bar
被展开为(*s.T).bar
, 而s.T
是个空指针,解引用会panic
.
可以使用下面的代码输出
s
:
func main() {
s := S{}
fmt.Printf("%#v", s) // 输出: main.S{T:(*main.T)(nil)}
}
2: 答案及解析: 锁失效. 将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()
}
「真诚赞赏,手留余香」
真诚赞赏,手留余香
使用微信扫描二维码完成支付
