登录
首页 >  Golang >  Go问答

奇异的len函数(或字符串)行为

来源:stackoverflow

时间:2024-03-01 18:09:25 243浏览 收藏

知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个Golang开发实战,手把手教大家学习《奇异的len函数(或字符串)行为》,在实现功能的过程中也带大家重新温习相关知识点,温故而知新,回头看看说不定又有不一样的感悟!

问题内容

我正在尝试使用 goquery 解析时间表内容以便稍后使用。但我有一个问题。

我有两个功能。第一个获取 html 文档并搜索令牌 (csrfmiddlewaretoken),第二个使用此令牌发送请求并提取信息。从页面中提取所有必要的信息后,我搜索令牌以在将来的请求中使用它并存储它。

但由于某种原因,found 令牌在到达 if len(foundcsrftoken) == 0 {.如果我在语句之前打印令牌的长度,它将打印以下内容:

...
64
0
...

我已经删除了所有 goroutine,以防万一出现问题。

func findcsrfmiddlewaretoken(responsebody io.reader) (string, error) {
    document, err := goquery.newdocumentfromreader(responsebody)
    if err != nil {
        return "", err
    }

    var foundcsrftoken string
    document.find("script").each(func(_ int, scrpt *goquery.selection) {
        scripttext := scrpt.text()
        if funcdefindex := strings.index(scripttext, "function filter"); funcdefindex != -1 {
            csrftokenvaluestart := strings.index(scripttext, "csrfmiddlewaretoken: '")
            offset := csrftokenvaluestart + len("csrfmiddlewaretoken: '")
            foundcsrftoken = scripttext[offset : offset+csrfmiddlewaretokenlength]
        }
    })
    if len(foundcsrftoken) == 0 {
        return "", errnocsrfmiddlewaretoken
    }
    return foundcsrftoken, nil
}

func (parser *TimetableParser) ParseTimetable(timetableFilterInfo internal.TimetableInfo) (internal.Timetable, error) {
    timetable := internal.Timetable{}

    requestBody := makeFormValues(timetableFilterInfo, parser.csrfMiddlewareToken).Encode()
    request, err := http.NewRequest("POST", baseUrl, strings.NewReader(requestBody))
    if err != nil {
        return timetable, err
    }
    request.Header.Add("Content-Type", "application/x-www-form-urlencoded")
    request.Header.Add("Content-Length", strconv.Itoa(len(requestBody)))
    request.Header.Add("Referer", baseUrl)

    response, err := parser.client.Do(request)
    if err != nil {
        return timetable, err
    }
    defer response.Body.Close()

    document, err := goquery.NewDocumentFromReader(response.Body)
    if err != nil {
        return timetable, err
    }

    document.Find("table#schedule").Find("tr").Each(func(rowIndex int, row *goquery.Selection) {
        subjectTimeElement := row.Closest("td")
        subjectTimeElement.NextAll().Each(func(columnIndex int, cell *goquery.Selection) {
            subjectInfo := extractSubjectInfoFromCell(cell)
            subjectInfo.Order = rowIndex
            timetable.Subjects[columnIndex][rowIndex] = subjectInfo
        })
    })

    parser.csrfMiddlewareToken, err = findCsrfMiddlewareToken(response.Body)
    if err != nil {
            log.Println("csrfMiddlewareToken: " + err.Error())
    }
    return timetable, nil
}

go版本:go1.17.1 windows/amd64

goquery 版本:1.7.1


正确答案


我刚刚意识到出了什么问题。 io.Reader 被视为流。所以当我读它一次时,它就变成空的了。正如您所看到的,在收集所有必要的信息并读取响应之后,它被传递到第一个函数中。但已经是空的了。 当我第一次调用 findCsrfMiddlewareToken 函数时,它照常工作并打印令牌长度(64)。但是当我进行第二次调用且响应为空时,它会打印 0。

可能的解决方案:How to read multiple times from same io.Reader

本篇关于《奇异的len函数(或字符串)行为》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>