go学习之文件读取问题(需更新)

go学习之文件读取问题(需更新)

 

一,问题:

今天做一个GO广度迷宫问题时,出现了一个奇怪的问题。

就是通过fmt.Fscanf()读取文件时,每行末尾多出一个0.

 

 

二,问题代码:

1,输入文件maze.in:

6 5
0 1 0 0 0
0 0 0 1 0
0 1 0 1 0
1 1 1 0 0
0 1 0 0 1
0 1 0 0 0

PS:第一行表示输入文件的row和col,下面的0表示可以通过,1表示无法通过。

 

2.读取文件函数readMaze:

func readMaze(filename string) [][]int {
    file, err := os.Open(filename)
    if err != nil {
        panic(err)
    }

    var row, col int
    fmt.Fscanf(file, "%d %d", &row, &col)

    maze := make([][]int, row)
    for i := range maze {
        maze[i] = make([]int, col)
        for j := range maze[i] {
            fmt.Fscanf(file, "%d", &maze[i][j])
        }
    }

    return maze
}

PS:这段代码时从网上ccoume那里剽过来的。

 

3.主函数main(只用于测试读取功能):

func main() {
    maze := readMaze("maze/maze.in")
    for _, row := range maze {
        for _, val := range row {
            fmt.Printf("%d ", val)
        }
        fmt.Println()
    }
}

 

 

三,问题代码执行的输出:

0 1 0 0 0 
0 0 0 1 0 
0 1 0 1 0 
1 1 1 0 0 
0 1 0 0 1 
0 1 0 0 0 

 

 

四,分析错误:

1.改变输入:

因为输入文件全是0与1,我们无法判断问题出在哪里,我就将输入文件改为下列方式:

6 5
1 2 3 4 5
6 7 8 9 10
11 12 13 14 15
16 17 18 19 20
21 22 23 24 25
26 27 28 29 30

同样是6行,5列,但是内容却是从1到30

 

2.测试输出:

0 1 2 3 4 
5 0 6 7 8 
9 10 0 11 12 
13 14 15 0 16 
17 18 19 20 0 
21 22 23 24 25 

 

 

3.分析问题:

我们可以看到在原来每次换行前都多了一个0,但是考虑到输入文件还有第一行的6和5,所以应该是每行换行后多了一个0.

并且,每个数字都有按顺序输入。代码中的file是连续的。而输出文件符合我们规定的数组形式。

所以代码本身没有错误。而只是原本输入文件的换行都有一个0,而fmt.Fscanf并没有处理。

 

 

五,解决方案:

1.改变输入文件:

可以通过将输入文件中所涉及到的换行都取消掉。这样就可以不用修改代码了。而且也可以确保在处理类似文件时,代码依旧不需要修改。

 

2.修改读取函数:

很多时候,我们无法修改输入文件,那么我们可以修改自己的读取函数readMaze()。

既然,fmt.Fscanf()无法跳过0,那我们就多读取一下0,但不输入到我们的读取结果数组中。

代码如下:

func readMaze(filename string) [][]int {
    file, err := os.Open(filename)
    if err != nil {
        panic(err)
    }

    var row, col int
    fmt.Fscanf(file, "%d %d", &row, &col)

    maze := make([][]int, row)
    for i := range maze {
        maze[i] = make([]int, col)
        fmt.Fscanf(file, "%d")
        //用以解决换行带来的0值问题。希望以后可以找到一个更好的解决方案。
        for j := range maze[i] {
            fmt.Fscanf(file, "%d", &maze[i][j])
        }
    }

    return maze
}

 

PS:按照代码的可读性以及逻辑来说,我应该在每次换行后添加

fmt.Fscanf(file, "%d")

即,应该写成如下代码:

func readMaze(filename string) [][]int {
    file, err := os.Open(filename)
    if err != nil {
        panic(err)
    }

    var row, col int
    fmt.Fscanf(file, "%d %d", &row, &col)
    fmt.Fscanf(file, "%d")

    maze := make([][]int, row)
    for i := range maze {
        maze[i] = make([]int, col)
        for j := range maze[i] {
            fmt.Fscanf(file, "%d", &maze[i][j])
        }
        fmt.Fscanf(file, "%d")
    }

    return maze
}

PS:这样更为友好,但我只是写一个小的广度迷宫读取,就偷懒了。。。

当然,前面读取5与6那两个fmt.Fsacnf是可以合起来,合为: 

fmt.Fscanf(file, "%d %d ", &row, &col)
//就是多了一个空格

 

 

3.自定义扩展fmt.Fscanf()函数:

之前的方法还是太有局限性了。不可能每次读取类似文件,我都加一句吧。

所以为了一劳永逸,完全可以写一个自定义的扩展函数。解决换行问题。

这里就不做演示。

 

 

六,求解更好解决方案:

这样的问题,肯定已经有了更为成熟的解决方案,只是作为小白的我没有找到,我在这里只是提供一个我自己想到的解决方法。

当然啦,更多的是一个找到问题,解决问题的路子吧。

如果有人知道更好的解决方案,可以告诉我。非常感谢。

另外,如果之后,我找到了更好的解决方法,我也会来更新的。

 

你可能感兴趣的:(go学习之文件读取问题(需更新))