继续啃《go程序设计语言》

//8、输出出现重复行的文件的名称
/*
//1、创建n个文件填入内容
//2、根据文件名打开文件
//2.1、读所有内容,如果有重复内容,将该文件名保存到slice中
//2.2、重复
//2.3、输出文件名
*/

func main() {
    //counts := make(map[string]int)
    var name []string
    files := os.Args[1:]
    if len(files) == 0 {
        return
    }
    for _, filename := range files {
        //fmt.Println(filename)
        f, err := os.Open(filename)
        if err != nil {
            fmt.Fprintf(os.Stderr, "dup2: %v\n", err)
            continue
        }
        counts := make(map[string]int) //counts每次循环都从新初始化,保证上一次循环的结果不会影响下一次循环
        counLines(f, counts)
        f.Close()
        for _, n := range counts {
            if n > 1 {
                name = append(name, filename)
                break
            }
        }
    }
    fmt.Println(strings.Join(name, " "))
}

func counLines(f *os.File, counts map[string]int) {
    input := bufio.NewScanner(f)
    for input.Scan() {
        counts[input.Text()]++
    }
}

在循环体外初始化的变量counts的影响分析:
原因分析:
1、第一次循环
1.1、执行代码打开test.txt(已知只有test.txt存在重复行,程序运行后终端应该只能输出test.txt)。
1.2、读取test.txt内容,对出现的每行内容出现次数进行计数,结果存入counts。
1.3、for _, n := range counts {}判断test.txt中存在重复行,将文件名保存到变量name中。
2、第二次循环
2.1、执行代码打开test1.txt。
2.2、读取test1.txt内容,对出现的每行内容出现次数进行计数,结果存入counts。test1.txt中存在test.txt中有的行,执行counts[input.Text()]++。
2.3、for _, n := range counts {}判断test1.txt中也存在重复行,将文件名保存到变量name中。
总结:
两次打开文件是独立事件,计数应该保持独立。
程序设计要进行步骤流程梳理。