15.3 访问并读取页面数据

在下边这个程序中,数组中的 url 都将被访问:会发送一个简单的 http.Head() 请求查看返回值;它的声明如下:func Head(url string) (r *Response, err error)

返回的响应 Response 其状态码会被打印出来。

示例 15.7 poll_url.go

  1. package main
  2. import (
  3. "fmt"
  4. "net/http"
  5. )
  6. var urls = []string{
  7. "http://www.google.com/",
  8. "http://golang.org/",
  9. "http://blog.golang.org/",
  10. }
  11. func main() {
  12. // Execute an HTTP HEAD request for all url's
  13. // and returns the HTTP status string or an error string.
  14. for _, url := range urls {
  15. resp, err := http.Head(url)
  16. if err != nil {
  17. fmt.Println("Error:", url, err)
  18. }
  19. fmt.Println(url, ": ", resp.Status)
  20. }
  21. }

输出为:

  1. http://www.google.com/ : 302 Found
  2. http://golang.org/ : 200 OK
  3. http://blog.golang.org/ : 200 OK

译者注 由于国内的网络环境现状,很有可能见到如下超时错误提示:

​ Error: http://www.google.com/ Head http://www.google.com/: dial tcp 216.58.221.100:80: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond.

在下边的程序中我们使用 http.Get() 获取并显示网页内容;Get() 返回值中的 Body 属性包含了网页内容,然后我们用 ioutil.ReadAll() 把它读出来:

示例 15.8 http_fetch.go

  1. package main
  2. import (
  3. "fmt"
  4. "io/ioutil"
  5. "log"
  6. "net/http"
  7. )
  8. func main() {
  9. res, err := http.Get("http://www.google.com")
  10. checkError(err)
  11. data, err := ioutil.ReadAll(res.Body)
  12. checkError(err)
  13. fmt.Printf("Got: %q", string(data))
  14. }
  15. func checkError(err error) {
  16. if err != nil {
  17. log.Fatalf("Get : %v", err)
  18. }
  19. }

当访问不存在的网站时,这里有一个 checkError 输出错误的例子:

  1. 2011/09/30 11:24:15 Get: Get http://www.google.bex: dial tcp www.google.bex:80:GetHostByName: No such host is known.

译者注 和上一个例子相似,你可以把 google.com 更换为一个国内可以顺畅访问的网址进行测试

在下边的程序中,我们获取一个 Twitter 用户的状态,通过 xml 包将这个状态解析成为一个结构:

示例 15.9 twitter_status.go

  1. package main
  2. import (
  3. "encoding/xml"
  4. "fmt"
  5. "net/http"
  6. )
  7. /*这个结构会保存解析后的返回数据。
  8. 他们会形成有层级的 XML,可以忽略一些无用的数据*/
  9. type Status struct {
  10. Text string
  11. }
  12. type User struct {
  13. XMLName xml.Name
  14. Status Status
  15. }
  16. func main() {
  17. // 发起请求查询推特 Goodland 用户的状态
  18. response, _ := http.Get("http://twitter.com/users/Googland.xml")
  19. // 初始化 XML 返回值的结构
  20. user := User{xml.Name{"", "user"}, Status{""}}
  21. // 将 XML 解析为我们的结构
  22. xml.Unmarshal(response.Body, &user)
  23. fmt.Printf("status: %s", user.Status.Text)
  24. }

输出:

  1. status: Robot cars invade California, on orders from Google: Google has been testing self-driving cars ... http://bit.ly/cbtpUN http://retwt.me/97p<exit code="0" msg="process exited normally"/>

译者注 和上边的示例相似,你可能无法获取到 xml 数据,另外由于 Go 版本的更新,xml.Unmarshal() 函数的第一个参数必需是 []byte 类型,而无法传入 Body

我们会在 15.4 节 中用到 http 包中的其他重要的函数:

  • http.Redirect(w ResponseWriter, r *Request, url string, code int):这个函数会让浏览器重定向到 url(可以是基于请求 url 的相对路径),同时指定状态码。
  • http.NotFound(w ResponseWriter, r *Request):这个函数将返回网页没有找到,HTTP 404 错误。
  • http.Error(w ResponseWriter, error string, code int):这个函数返回特定的错误信息和 HTTP 代码。
  • 另一个 http.Request 对象 req 的重要属性:req.Method,这是一个包含 GETPOST 字符串,用来描述网页是以何种方式被请求的。

Go 为所有的 HTTP 状态码定义了常量,比如:

  1. http.StatusContinue = 100
  2. http.StatusOK = 200
  3. http.StatusFound = 302
  4. http.StatusBadRequest = 400
  5. http.StatusUnauthorized = 401
  6. http.StatusForbidden = 403
  7. http.StatusNotFound = 404
  8. http.StatusInternalServerError = 500

你可以使用 w.header().Set("Content-Type", "../..") 设置头信息。

比如在网页应用发送 html 字符串的时候,在输出之前执行 w.Header().Set("Content-Type", "text/html")(通常不是必要的)。

练习 15.4:扩展 http_fetch.go 使之可以从控制台读取 url,使用 12.1 节学到的接收控制台输入的方法 (http_fetch2.go)

练习 15.5:获取 json 格式的推特状态,就像示例 15.9 (twitter_status_json.go)

链接