1. 第六十天
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
}
参考答案及解析:第 19 行,因为 s.bar 将被展开为 (*s.T).bar
,而 s.T 是个空指针,解引用会 panic。
可以使用下面代码输出 s:
func main() {
s := S{}
fmt.Printf("%#v",s) // 输出:main.S{T:(*main.T)(nil)}
}
引自:《Go语言101》
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()
}
参考答案及解析:锁失效。将 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()
}
引自:《Go 语言学习笔记》· 同步