写过其他后端语言的人,在写golang的时候,总会遇到一些奇怪的问题,而且很难排查,我个人觉得是因为两个原因,第一个golang基础不扎实,第二个编程思维没有转换过来。下面都是我自己遇到或者想到的容易遇到的点。如果转载的话,请注明出处,因为我会不定期更新。
1、到底是什么类型
这是我自己想的问题,看以下代码,T到底是什么类型?
type A interface{
Do() error
}
type B interface{
Do() error
}
type T int
func (t T)Do() error{
return nil
}
曾经在某个golang群里问过这个问题,当时没有人回答这个问题,也许是太简单了,不屑,或者是不确定。
请认真思考。
下面是我的分析
T到底什么类型?首先万物皆interface,其次,看T的定义,是int的别名,也就是int32的别名,最后,T实现了Do()方法,但是,A 和 B都申明了这个方法,那么,T到底是A类型呢,还是B类型?答案是T既是
A也是B。为什么?因为golang中,只要实现了某个接口的所有方法,那么它就是这种类型。另一个佐证是goland中的实现关系:
2、range迭代
代码如下,请先思考运行后的结果是什么:
type student struct {
Name string
Age int
}
func main (){
m := make(map[string]*student)
stus := []student{
{Name: "zhou", Age: 24},
{Name: "li", Age: 23},
{Name: "wang", Age: 22},
}
for _, stu := range stus {
m[stu.Name] = &stu
}
for _,stu := range stus{
fmt.Println(m[stu.Name])
}
}
是不是觉得应该是:
&{zhou 24}
&{li 23}
&{wang 22}
但,运行之后的结果为:
&{wang 22}
&{wang 22}
&{wang 22}
为什么?首先从短赋值说起
for _, stu := range stus {
m[stu.Name] = &stu
}
等同与
var stu student
for _, stu = range stus {
m[stu.Name] = &stu
}
是不是有点头绪了?因为每次m[stu.Name] = &stu 执行的时候,指向都是stu,而每次迭代,stu指向不同的student,最终的结果就是,所有的数据都指向了最后一个student。至于stu.Name 是不用怀疑的,每次迭代都是立即求值。
那么如何解决呢?既然问题是由于stu引起的,那么,每次迭代让指针指向迭代对象就可以解决,引入新变量就可以:
for _, stu := range stus {
st := stu
m[st.Name] = &st
}
或者,map的值不存指针,存对象即可:
m := make(map[string]*student)
...
for _, stu := range stus {
m[stu.Name] = stu
}
...
3、申明类型
这个是我写工具类的时候遇到的,不多说,直接上代码:
var buf bytes.Buffer
...//write data to buf
r, err := http.NewRequest("PUT", uri+"/apis/", buf)
你会发现,最后一行会报错的,看下NewRequest的定义,最后一个参数是io.Reader类型:
func NewRequest(method, url string, body io.Reader) (*Request, error)
func (b *Buffer) Read(p []byte) (n int, err error)
很明显,bytes.Buffer 实现了io.Reader接口,根据第一个问题得出的结论,Buffer就是io.Reader类型啊,为什么会报错呢?当然,也不能申明buf为io.Reader对吧,因为我们要往buf里面写入数据呢。
当然用Buffer是没错啦,问题在于我们显式申明buf 为bytes.Buffer,那么类型就已经定了,注定不能用在io.Reader参数上了,虽然它确实是实现了io.Reader 的接口。
原因找到了,怎么解决呢?我们可以不用申明buf是Buffer类型:
buf := new(bytes.Buffer)
...//write data to buf
r, err := http.NewRequest("PUT", uri+"/apis/", buf)
这样就OK了。