«Неверный адрес памяти» в Голанге с использованием ioutil.ReadAll () - PullRequest
0 голосов
/ 26 декабря 2018

В настоящее время я изучаю Голанг (и пока мне это нравится).Но, к сожалению, я застрял на пару часов и, похоже, я не нашел решения своей проблемы в Google.

Так вот моя проблема.У меня есть этот фрагмент кода (из учебника):

func main() {
    var s SitemapIndex

    resp, _ := http.Get("https://www.washingtonpost.com/news-sitemaps/index.xml")
    bytes, _ := ioutil.ReadAll(resp.Body)
    resp.Body.Close()

    xml.Unmarshal(bytes, &s)

    for _, Location := range s.Locations {
        resp, _ := http.Get(Location)
        ioutil.ReadAll(resp.Body)

    }

}

Я знаю, что мой код неполон, но это потому, что я удалил части, которые не вызывали проблему, чтобы сделать его более читабельным в Stackoverflow.

Поэтому, когда я получаю содержимое Location и пытаюсь обработать данные с помощью ioutil.ReadAll(), я получаю эту ошибку с указанием указателя:

panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x1 addr=0x40 pc=0x1210a69]

goroutine 1 [running]:
main.main()
    /Users/tom/Developer/Go/src/news/index.go:23 +0x159
exit status 2

Я действительно не понимаю эту ошибкунезависимо от того, сколько я смотрю на это.Я пытался извлечь ошибку из ioutil.ReadAll(resp.Body), выполнив _, e := ioutil.ReadAll(resp.Body), а затем напечатав e, но при этом выдает еще одну ошибку ...

Я где-то читал, что это может бытьпотому что тело, возвращенное мне, содержит ошибки, но в учебнике оно работает нормально.

Надеюсь, у вас, ребята, будет решение для меня.Спасибо.

РЕДАКТИРОВАТЬ: Вот структуры, которые я определил:

type SitemapIndex struct {
    Locations []string `xml:"sitemap>loc"`
}

type News struct {
    Titles []string `xml:"url>news>title"`
    Keywords []string `xml:"url>news>keywords"`
    Locations []string `xml:"url>loc"`
}

type NewsMap struct {
    Keyword string
    Location string
}

Ответы [ 3 ]

0 голосов
/ 26 декабря 2018

Первое правило Go: проверка на наличие ошибок.


Когда вызов функции возвращает ошибку, вызывающая сторона несет ответственность за ее проверку и принятие соответствующих действий..

Обычно, когда функция возвращает ненулевую ошибку, другие ее результаты не определены и должны игнорироваться.

Язык программирования Go, Алан А. А. Донован и Брайан В. Керниган


Например,

if err != nil {
    fmt.Printf("%q\n", Location) // debug error
    fmt.Println(resp)            // debug error
    fmt.Println(err)
    return
}

Вывод:

"\nhttps://www.washingtonpost.com/news-sitemaps/politics.xml\n"
<nil>
parse 
https://www.washingtonpost.com/news-sitemaps/politics.xml
: first path segment in URL cannot contain colon

Если вы не уловили эту ошибку и продолжаете работать сresp == nil затем

bytes, err := ioutil.ReadAll(resp.Body)

Вывод:

panic: runtime error: invalid memory address or nil pointer dereference

package main

import (
    "encoding/xml"
    "fmt"
    "io/ioutil"
    "net/http"
    "strings"
)

type SitemapIndex struct {
    Locations []string `xml:"sitemap>loc"`
}

func main() {
    var s SitemapIndex

    resp, err := http.Get("https://www.washingtonpost.com/news-sitemaps/index.xml")
    if err != nil {
        fmt.Println(err)
        return
    }
    bytes, err := ioutil.ReadAll(resp.Body)
    if err != nil {
        fmt.Println(err)
        return
    }
    err = resp.Body.Close()
    if err != nil {
        fmt.Println(err)
        return
    }

    err = xml.Unmarshal(bytes, &s)
    if err != nil {
        fmt.Println(err)
        return
    }

    for _, Location := range s.Locations {
        resp, err := http.Get(Location)
        if err != nil {
            fmt.Printf("%q\n", Location) // debug error
            fmt.Println(resp)            // debug error
            fmt.Println(err)
            return
        }
        bytes, err := ioutil.ReadAll(resp.Body)
        if err != nil {
            fmt.Println(err)
            return
        }
        fmt.Println(len(bytes))
        err = resp.Body.Close()
        if err != nil {
            fmt.Println(err)
            return
        }
    }
}
0 голосов
/ 26 декабря 2018

Хорошо, я нашел причину своей проблемы и способы ее решения.Проблема заключалась в том, что в URL были некоторые символы новой строки, которых я не видел.

Поэтому вместо этого:

resp, err := http.Get(Location)

Я сделал это:

resp, err := http.Get(strings.TrimSpace(Location))

Эторешил это.

0 голосов
/ 26 декабря 2018

Как упоминал Мостафа, вы должны правильно обрабатывать ошибки.Там нет попытки поймать в Голанге.Как то так я устал.по крайней мере это ловит ошибку URL в liteIde

package main

import (
    "encoding/xml"
    "fmt"
    "io/ioutil"
    "net/http"
    "os"
    "runtime"
)

type SitemapIndex struct {
    Locations []string `xml:"sitemap>loc"`
}

func main() {
    var s SitemapIndex

    resp, err := http.Get("https://www.washingtonpost.com/news-sitemaps/index.xml")
    if err != nil {
        fmt.Println("Unable to get the url ")
        os.Exit(1)
    }
    bytes, _ := ioutil.ReadAll(resp.Body)
    defer resp.Body.Close()
    //fmt.Println(string(bytes))
    xml.Unmarshal(bytes, &s)
    //fmt.Println(len(s.Locations))

    for _, Location := range s.Locations {
        //fmt.Println(Location)
        go func() {
            r, err := http.Get(Location)
            if err != nil {
                fmt.Println("Error occured ", err)
                return
            }
            bytes, err := ioutil.ReadAll(resp.Body)
            if err != nil {
                fmt.Printf("Error in reading :", err)
            }
            fmt.Println(string(bytes))
            r.Body.Close()
        }()
    }
    runtime.Gosched()
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...