
当需要从一个包含浮点数的字符串中逐个读取数据时,fmt包提供了一系列扫描函数。初学者可能会倾向于使用fmt.fscanf,因为它允许指定格式字符串,这在很多c语言背景的开发者看来是直观的。然而,fmt.fscanf在处理输入中的换行符时,行为可能不如预期。
考虑以下示例代码,它尝试使用fmt.Fscanf从一个包含换行符的字符串中解析浮点数:
package main
import (
"fmt"
"strings"
)
func main() {
var z float64
var a []float64
// 包含换行符的字符串
s := "3.25 -12.6 33.7 \n 3.47"
in := strings.NewReader(s)
fmt.Println("使用 fmt.Fscanf 解析:")
for {
n, err := fmt.Fscanf(in, "%f", &z)
fmt.Printf("扫描结果: n=%d, 值=%.2f, 错误=%v\n", n, z, err)
if err != nil {
break
}
a = append(a, z)
}
fmt.Println("解析到的浮点数数组:", a)
// 重置读取器以便后续测试
in = strings.NewReader(s)
a = nil // 清空数组
}运行上述代码,输出如下:
使用 fmt.Fscanf 解析: 扫描结果: n=1, 值=3.25, 错误=<nil> 扫描结果: n=1, 值=-12.60, 错误=<nil> 扫描结果: n=1, 值=33.70, 错误=<nil> 扫描结果: n=0, 值=0.00, 错误=EOF 解析到的浮点数数组: [3.25 -12.6 33.7]
从输出可以看出,fmt.Fscanf成功解析了前三个浮点数,但在遇到换行符\n之后,它停止了进一步的解析,并最终返回了EOF错误。这是因为fmt.Fscanf在默认情况下,会将格式字符串中的空白字符(如%f前的空格)与输入中的空白字符(包括空格、制表符、换行符)进行匹配。但是,当格式字符串中没有明确指定匹配换行符时,fmt.Fscanf会将换行符视为非匹配字符,从而导致解析中断。它期望输入中的换行符与格式字符串中的换行符精确匹配。
对于上述场景,更简洁有效的解决方案是使用fmt.Fscan。与fmt.Fscanf不同,fmt.Fscan在处理输入时,会将所有空白字符(包括空格、制表符和换行符)都视为字段分隔符。这意味着它会跳过这些空白字符,直到找到下一个非空白字符,并尝试将其解析为指定类型。
立即学习“go语言免费学习笔记(深入)”;
修改后的代码如下:
package main
import (
"fmt"
"strings"
"io" // 导入io包以使用io.EOF
)
func main() {
var z float64
var a []float64
s := "3.25 -12.6 33.7 \n 3.47"
in := strings.NewReader(s)
fmt.Println("使用 fmt.Fscan 解析:")
for {
n, err := fmt.Fscan(in, &z) // 注意这里是 fmt.Fscan
fmt.Printf("扫描结果: n=%d, 值=%.2f, 错误=%v\n", n, z, err)
if err == io.EOF { // 当遇到文件末尾时退出循环
break
}
if err != nil { // 处理其他错误
fmt.Printf("解析过程中发生错误: %v\n", err)
break
}
a = append(a, z)
}
fmt.Println("解析到的浮点数数组:", a)
}运行上述代码,输出如下:
使用 fmt.Fscan 解析: 扫描结果: n=1, 值=3.25, 错误=<nil> 扫描结果: n=1, 值=-12.60, 错误=<nil> 扫描结果: n=1, 值=33.70, 错误=<nil> 扫描结果: n=1, 值=3.47, 错误=<nil> 解析到的浮点数数组: [3.25 -12.6 33.7 3.47]
可以看到,fmt.Fscan成功解析了字符串中的所有浮点数,即使它们被换行符分隔。
| 特性 | fmt.Fscan(r io.Reader, a ...interface{}) | fmt.Fscanf(r io.Reader, format string, a ...interface{}) |
|---|---|---|
| 空白字符处理 | 将所有空白字符(空格、制表符、换行符)视为字段分隔符,并跳过它们。 | 严格按照format字符串匹配。如果format中没有明确的空白字符匹配(如\n),则输入中的换行符可能导致不匹配或解析中断。 |
| 格式控制 | 不支持格式字符串,仅根据变量类型进行默认解析。 | 支持格式字符串(如%f, %s, %d, %5f等),提供更精细的输入格式控制。 |
| 适用场景 | 适用于输入数据由任意空白字符(包括换行符)分隔的简单场景。 | 适用于需要严格匹配特定输入格式(如固定宽度字段、特定分隔符)的复杂场景。 |
尽管fmt.Fscan在处理由空白字符分隔的数据时更为便捷,fmt.Fscanf在某些特定场景下仍然是不可替代的:
注意事项: 如果在fmt.Fscanf的格式字符串中需要匹配换行符,你必须显式地在格式字符串中包含\n。例如,fmt.Fscanf(in, "%f\n", &z) 会尝试读取一个浮点数,然后严格匹配一个换行符。这通常会使代码变得复杂,因为你需要处理输入中可能存在或不存在的换行符。
在Go语言中从字符串读取浮点数时,选择fmt.Fscan还是fmt.Fscanf取决于你的输入数据的结构和解析需求:
理解这两个函数的细微差别,能帮助开发者编写出更高效、更准确的Go语言数据解析代码。在大多数从字符串中读取以空格或换行符分隔的数值的场景中,fmt.Fscan通常是更简单、更可靠的选择。
以上就是Go语言中从字符串解析浮点数:Fscan 与 Fscanf 的选择与实践的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号