常问面试题
  - 函数返回return XX, 是原子操作吗?(不是, 如果你看过雨痕老师的内存管理,就知道)
- 普通函数和闭包函数的区别有哪些?后者的应用场景在哪里?(闭包函数使用外部变量,是引用传递,注意:闭包函数运行时使用到该变量时,获取当时该变量的值。应用场景:闭包函数要比较小,有点像内联函数,用于defer等其他场景-关闭连接等)
- make和new有什么区别(make用于channel、map和slice的创建并初始化零值。new是指针对象。Effective Go中有个例子)
var p *[]int = new([]int)       // p=nil, 没有任何用处,只是分配了一个指针对象 var v  []int = make([]int, 100) // 而make则分配了长度和容量等于100,并且初始化零值的对象  // 复杂方法: var p *[]int = new([]int) *p = make([]int, 100, 100)  // 常用方法: v := make([]int, 100) 
 查找重复行
 代码片段
 func findDup() {     scanner:=bufio.NewScanner(os.Stdin)     for scanner.Scan(){         maps[scanner.Text()]++     }     for line, count := range maps{         if count >0 {             fmt.Printf("line: %s, count: %d\n",                 line, count)         }     } } 
 bufio是golang标准库,表示缓冲IO的意思,可读可写。通过bufio标准库,实现对输入数据的读取、处理和输出操作
 我们可以由此了解并使用bufio库中常用的方法:
 bufio实现了io.Reader和io.Writer接口, 所以可读可写。
 bufio.Reader作用是把oldReader的数据读取到缓冲区,它的常用方法列表
 reader:=bufio.NewReader(oldReader) // 默认size为4096 reader:=bufio.NewReaderSize(oldReader, size)  bytes, error = reader.ReadSlice(delim/*结束符*/) bytes, error = reader.ReadString(delim) bytes, error = reader.ReadLine() // 默认换行符\n bytes, error = reader.ReadBytes(delim) n, error = reader.Reader([]byte) 
 bufio.Reader使用DEMO:
 reader:=bufio.NewReader(strings.NewReader("ABCDEFG\nHIJKLMN\n") var bts = make([]byte, 100) // 注意点:需要初始化 _, err = reader.Read(bts) 
 这里要平时注意的一点:var bts = make([]byte, 100), 需要初始化。slice类型变量首先要初始化, 如果不初始化,则如果在调用的方法体内初始化,则返回时,slice类型变量还是旧值。原因说明:
  - slice底层实际上是struct结构体类型,有len,cap和指针数组组成。所以函数体内初始化则内存地址和传入的内存地址不是一致的,读到的数据是旧的
- 如果传入参数是初始化的,则相当于func(temp=Struct{}/传入参数param/), 虽然地址不一样,但是重新赋值了,且len、cap值相同,底层的指针数组还是指向一样的内存地址,如果底层数组的内存地址存放的数据发生变化,由于指向数组的指针没有变化,则传入参数可是指向了底层数组新的数据。
结论:所以凡是遇到slice类型变量,作为参数传入时,都需要进行make初始化,且指定slice的长度len。