فهرست منبع

Merge pull request #191 from glight2000/master

增加15.3.md 修改15.2.md下一节连接名
Unknwon 10 سال پیش
والد
کامیت
6da7950a17
4فایلهای تغییر یافته به همراه180 افزوده شده و 1 حذف شده
  1. 1 0
      README_gc.md
  2. 1 1
      eBook/15.2.md
  3. 148 0
      eBook/15.3.md
  4. 30 0
      eBook/examples/chapter_15/http_fetch2.go

+ 1 - 0
README_gc.md

@@ -33,3 +33,4 @@ Golang 编程:245386165
 |----------|------------------
 |2015-11-25|13.10 性能调试:分析并优化 Go 程序
 |2015-12-10|15.0 15.1 15.2
+|2015-12-12|15.3

+ 1 - 1
eBook/15.2.md

@@ -108,4 +108,4 @@ func (obj *Typ) ServeHTTP(w http.ResponseWriter, req *http.Request) {
 
 - [目录](directory.md)
 - 上一章:[tcp服务器](15.1.md)
-- 下一节:[写一个简单的网页应用](15.3.md)
+- 下一节:[访问并读取页面数据](15.3.md)

+ 148 - 0
eBook/15.3.md

@@ -0,0 +1,148 @@
+# 15.3 访问并读取页面
+
+在下边这个程序中,数组中的url都将被访问:会发送一个简单的`http.Head()`请求查看返回值;它的声明如下:`func Head(url string) (r *Response, err error)`
+
+返回状态码会被打印出来。
+
+示例 15.7 [poll_url.go](examples/chapter_15/poll_url.go):
+```go
+package main
+
+import (
+	"fmt"
+	"net/http"
+)
+
+var urls = []string{
+	"http://www.google.com/",
+	"http://golang.org/",
+	"http://blog.golang.org/",
+}
+
+func main() {
+	// Execute an HTTP HEAD request for all url's
+	// and returns the HTTP status string or an error string.
+	for _, url := range urls {
+		resp, err := http.Head(url)
+		if err != nil {
+			fmt.Println("Error:", url, err)
+		}
+		fmt.Print(url, ": ", resp.Status)
+	}
+}
+```
+输出为:
+```
+http://www.google.com/ : 302 Found
+http://golang.org/ : 200 OK
+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 pa
+rty did not properly respond after a period of time, or established connection failed because connected host has failed to respond.
+```
+在下边的程序中我们使用`http.Get()`获取网页内容; `Get`的返回值`res`中的`Body`属性包含了网页内容,然后我们用`ioutil.ReadAll`把它读出来:
+
+示例 15.8 [http_fetch.go](examples/chapter_15/http_fetch.go):
+```go
+package main
+
+import (
+	"fmt"
+	"io/ioutil"
+	"log"
+	"net/http"
+)
+
+func main() {
+	res, err := http.Get("http://www.google.com")
+	checkError(err)
+	data, err := ioutil.ReadAll(res.Body)
+	checkError(err)
+	fmt.Printf("Got: %q", string(data))
+}
+
+func checkError(err error) {
+	if err != nil {
+		log.Fatalf("Get : %v", err)
+	}
+}
+```
+
+当访问不存在的网站时,这里有一个`CheckError`输出错误的例子:
+```
+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](examples/chapter_15/twitter_status.go)
+```go
+package main
+
+import (
+	"encoding/xml"
+	"fmt"
+	"net/http"
+)
+
+/*这个结构会保存解析后的返回数据。
+他们会形成有层级的XML,可以忽略一些无用的数据*/
+type Status struct {
+	Text string
+}
+
+type User struct {
+	XMLName xml.Name
+	Status  Status
+}
+
+func main() {
+	// 发起请求查询推特Goodland用户的状态
+	response, _ := http.Get("http://twitter.com/users/Googland.xml")
+	// 初始化XML返回值的结构
+	user := User{xml.Name{"", "user"}, Status{""}}
+	// 将XML解析为我们的结构
+	xml.Unmarshal(response.Body, &user)
+	fmt.Printf("status: %s", user.Status.Text)
+}
+```
+输出:
+```
+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](15.4.md)中用到`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`,这是一个包含`GET`或`POST`字符串,用来描述网页是以何种方式被请求的。
+
+go为所有的HTTP状态码定义了常量,比如:
+
+	http.StatusContinue		= 100
+	http.StatusOK			= 200
+	http.StatusFound		= 302
+	http.StatusBadRequest		= 400
+	http.StatusUnauthorized		= 401
+	http.StatusForbidden		= 403
+	http.StatusNotFound		= 404
+	http.StatusInternalServerError	= 500
+
+你可以使用`w.header().Set("Content-Type", "../..")设置头信息
+
+比如在网页应用发送html字符串的时候,在输出之前执行`w.Header().Set("Content-Type", "text/html")`。
+
+练习 15.4:扩展 http_fetch.go 使之可以从控制台读取url,使用[章节12.1](12.1.md)学到的接收控制台输入的方法 ([http_fetch2.go](examples/chapter_15/http_fetch2.go))
+
+练习 15.5:获取json格式的推特状态,就像示例 15.9(twitter_status_json.go)
+
+
+## 链接
+
+- [目录](directory.md)
+- 上一章:[一个简单的网页服务器](15.2.md)
+- 下一节:[写一个简单的网页应用](15.4.md)

+ 30 - 0
eBook/examples/chapter_15/http_fetch2.go

@@ -0,0 +1,30 @@
+package main
+
+import (
+	"bufio"
+	"fmt"
+	"io/ioutil"
+	"log"
+	"net/http"
+	"os"
+	"strings"
+)
+
+func main() {
+	inputReader := bufio.NewReader(os.Stdin)
+	fmt.Println("Please enter the url...")
+	url, err := inputReader.ReadString('\n')
+	url = strings.TrimSuffix(url, "\r\n")
+	checkError(err)
+	res, err := http.Get(url)
+	checkError(err)
+	data, err := ioutil.ReadAll(res.Body)
+	checkError(err)
+	fmt.Printf("Got: %q", string(data))
+}
+
+func checkError(err error) {
+	if err != nil {
+		log.Fatalf("Get : %v", err)
+	}
+}