当我使用 fmt.Fscanf 时,为什么会出现“输入与格式不匹配”​​的情况?

我正在尝试使用fmt.Fscanf,但我很难弄清楚如何使用。有以下代码:


package main


import (

    "fmt"

    "strings"

)


func main() {

    var host, user, date, httpStr string

    var code, size int


    r := strings.NewReader(`127.0.0.1 - james [09/May/2018:16:00:39 +0000] "GET /report HTTP/1.0" 200 123`)


    _, err := fmt.Fscanf(r, `%s - %s [%s], "%s" %d %d`,

        &host, &user, &date, &httpStr, &code, &size)

    if err != nil {

        fmt.Printf("Failed to parse log line, error: %+v\n", err)

        panic(err)

    }


    fmt.Println(host, user, date, httpStr, code, size)


}

去游乐场:https://play.golang.org/p/zGxc6MXOF3a


我得到:


Failed to parse log line, error: input does not match format

panic: input does not match format


goroutine 1 [running]:

main.main()

        .../fscanf/main.go:19 +0x57b

exit status 2

为什么?


达令说
浏览 144回答 1
1回答

慕工程0101907

该错误来自如何Fscanf解析空格分隔的字符串。读取日期和 HTTP 字符串时,这会成为一个问题:当它读取日期时,09/May/2018:16:00:39 +0000它不会读取日期,而是会读取直到第一个空格:09/May/2018:16:00:39然后是expect ]。要解决此问题,将字符串解析为 是有意义的[%s %s],但这会更巧妙地失败。它将解析时区,+0000]因为它停在第一个空格处,而不是根据格式字符串。HTTP 字符串也会出现同样的问题:它在动词之后停止解析GET,然后在路由之后停止/report解析,但是在读取协议时它读取HTTP/1.0"not HTTP/1.0。一个可行的解决方案如下:package mainimport (    "fmt"    "log"    "strings")func main() {    var host, user, date, tzOffset, verb, route, proto string    var code, size int    r := strings.NewReader(`127.0.0.1 - james [09/May/2018:16:00:39 +0000] "GET /report HTTP/1.0" 200 123`)    _, err := fmt.Fscanf(r, `%s - %s %s %s %s %s %s %d %d`,        &host, &user, &date, &tzOffset, &verb, &route, &proto, &code,        &size)    if err != nil {        log.Fatal(err)    }    date = date[1:] + " " + tzOffset[:len(tzOffset)-1]    httpString := verb[1:] + " " + route + " " + proto[:len(proto)-1]    fmt.Println(host, user, date, httpString, code, size)}这里,我们在解析时忽略日期和 HTTP 字符串周围的括号和引号,然后将其删除。输出:127.0.0.1 james 09/May/2018:16:00:39 +0000 GET /report HTTP/1.0 200 123
打开App,查看更多内容
随时随地看视频慕课网APP

相关分类

Go