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()函数:
之前的方法还是太有局限性了。不可能每次读取类似文件,我都加一句吧。
所以为了一劳永逸,完全可以写一个自定义的扩展函数。解决换行问题。
这里就不做演示。
六,求解更好解决方案:
这样的问题,肯定已经有了更为成熟的解决方案,只是作为小白的我没有找到,我在这里只是提供一个我自己想到的解决方法。
当然啦,更多的是一个找到问题,解决问题的路子吧。
如果有人知道更好的解决方案,可以告诉我。非常感谢。
另外,如果之后,我找到了更好的解决方法,我也会来更新的。