回到简体

This commit is contained in:
chai2010
2016-02-15 11:06:34 +08:00
parent 9e878f9944
commit 2b37b23285
177 changed files with 2354 additions and 2354 deletions

View File

@@ -1,6 +1,6 @@
## 1.7. Web服
## 1.7. Web服
Go言的內置庫使得寫一個類似fetch的web服務器變得異常地簡單。在本中,我們會展示一微型服器,這個服務器的功能是返迴當前用正在訪問的URL。比如用戶訪問的是 http://localhost:8000/hello ,那麽響應是URL.Path = "hello"。
Go言的内置库使得写一个类似fetch的web服务器变得异常地简单。在本中,我们会展示一微型服器,这个服务器的功能是返回当前用正在访问的URL。比如用户访问的是 http://localhost:8000/hello ,那么响应是URL.Path = "hello"。
<u><i>gopl.io/ch1/server1</i></u>
```go
@@ -24,15 +24,15 @@ func handler(w http.ResponseWriter, r *http.Request) {
}
```
們隻用了八九行代碼就實現了一Web服程序,都是多虧了標準庫里的方法已經幫我們完成了大量工作。main函數將所有送到/路下的求和handler函數關聯起來,/開頭的請求其就是所有送到前站上的求,服務監聽8000端口。送到這個服務的“求”是一http.Request型的象,這個對象中包含了求中的一列相字段,其中就包括我需要的URL。當請求到達服務器時這個請求會被傳給handler函數來處理這個函數會將/hello這個路徑從請求的URL中解析出,然把其送到響應中,里我用的是標準輸出流的fmt.Fprintf。Web服務會在第7.7中做更詳細的闡述。
们只用了八九行代码就实现了一Web服程序,都是多亏了标准库里的方法已经帮我们完成了大量工作。main函数将所有送到/路下的求和handler函数关联起来,/开头的请求其就是所有送到前站上的求,服务监听8000端口。送到这个服务的“求”是一http.Request型的象,这个对象中包含了求中的一列相字段,其中就包括我需要的URL。当请求到达服务器时这个请求会被传给handler函数来处理这个函数会将/hello这个路径从请求的URL中解析出,然把其送到响应中,里我用的是标准输出流的fmt.Fprintf。Web服务会在第7.7中做更详细的阐述。
讓我們在後台運行這個服務程序。如果你的操作繫統是Mac OS X或者Linux麽在運行命令的末尾加上一&號,卽可讓程序簡單地跑在windows下可以在另外一命令行窗口去運行這個程序。
让我们在后台运行这个服务程序。如果你的操作系统是Mac OS X或者Linux么在运行命令的末尾加上一&号,即可让程序简单地跑在windows下可以在另外一命令行窗口去运行这个程序。
```
$ go run src/gopl.io/ch1/server1/main.go &
```
在可以通命令行來發送客戶端請求了:
在可以通命令行来发送客户端请求了:
```
$ go build gopl.io/ch1/fetch
@@ -42,11 +42,11 @@ $ ./fetch http://localhost:8000/help
URL.Path = "/help"
```
可以直接在瀏覽器里訪問這個URL得到返迴結果,如1.2
可以直接在浏览器里访问这个URL得到返回结果,如1.2
![](../images/ch1-02.png)
這個服務的基礎上疊加特性是很容易的。一種比較實用的改是爲訪問的url添加某種狀態。比如,下面這個版本出了同樣的內容,但是會對請求的次數進行計算;URL的請求結果會包含各URL被訪問的總次數,直接/count這個URL的訪問要除外。
这个服务的基础上叠加特性是很容易的。一种比较实用的改是为访问的url添加某种状态。比如,下面这个版本出了同样的内容,但是会对请求的次数进行计算;URL的请求结果会包含各URL被访问的总次数,直接/count这个URL的访问要除外。
<u><i>gopl.io/ch1/server2</i></u>
```go
@@ -85,9 +85,9 @@ func counter(w http.ResponseWriter, r *http.Request) {
}
```
這個服務器有兩個請求處理函,根據請求的url不同會調用不同的函數:對/count這個url的請求會調用到count這個函數其它的url都會調用默認的處理函。如果你的求pattern是以/尾,那所有以url爲前綴的url都會被這條規則匹配。在些代的背,服器每一次接收請求處理時都會另起一goroutine這樣服務器就可以同一時間處理多個請求。然而在併發情況下,假如的有兩個請求同一刻去更新count麽這個值可能併不會被正地增加;這個程序可能會引發一個嚴重的bug競態條件參見9.1)。了避免這個問題,我們必須保證每次脩改變量的最多能有一goroutine也就是代里的mu.Lock()和mu.Unlock()調用將脩改count的所有行包在中的目的。第九章中我們會進一步解共享量。
这个服务器有两个请求处理函,根据请求的url不同会调用不同的函数:对/count这个url的请求会调用到count这个函数其它的url都会调用默认的处理函。如果你的求pattern是以/尾,那所有以url为前缀的url都会被这条规则匹配。在些代的背,服器每一次接收请求处理时都会另起一goroutine这样服务器就可以同一时间处理多个请求。然而在并发情况下,假如的有两个请求同一刻去更新count么这个值可能并不会被正地增加;这个程序可能会引发一个严重的bug竞态条件参见9.1)。了避免这个问题,我们必须保证每次修改变量的最多能有一goroutine也就是代里的mu.Lock()和mu.Unlock()调用将修改count的所有行包在中的目的。第九章中我们会进一步解共享量。
下面是一個更爲豐富的例子handler函數會把請求的http頭和請求的form數據都打印出來,這樣可以使檢査和調試這個服務更爲方便:
下面是一个更为丰富的例子handler函数会把请求的http头和请求的form数据都打印出来,这样可以使检查和调试这个服务更为方便:
<u><i>gopl.io/ch1/server3</i></u>
```go
@@ -108,7 +108,7 @@ func handler(w http.ResponseWriter, r *http.Request) {
}
```
用http.Request這個struct里的字段來輸出下面這樣的內容:
用http.Request这个struct里的字段来输出下面这样的内容:
```
GET /?q=query HTTP/1.1
@@ -119,7 +119,7 @@ RemoteAddr = "127.0.0.1:59911"
Form["q"] = ["query"]
```
可以看到里的ParseForm被嵌套在了if句中。Go言允許這樣的一個簡單的語句結果作爲循環的變量聲明出在if句的最前面,這一點對錯誤處理很有用。我們還可以像下面這樣寫(當然看起來就長了一些):
可以看到里的ParseForm被嵌套在了if句中。Go言允许这样的一个简单的语句结果作为循环的变量声明出在if句的最前面,这一点对错误处理很有用。我们还可以像下面这样写(当然看起来就长了一些):
```go
err := r.ParseForm()
@@ -128,13 +128,13 @@ if err != nil {
}
```
用if和ParseForm合可以讓代碼更加簡單,併且可以限err這個變量的作用域,這麽做是很不的。我們會在2.7節中講解作用域。
用if和ParseForm合可以让代码更加简单,并且可以限err这个变量的作用域,这么做是很不的。我们会在2.7节中讲解作用域。
些程序中,我看到了很多不同的型被出到標準輸出流中。比如前面的fetch程序把HTTP的響應數據拷貝到了os.Stdoutlissajous程序里我們輸出的是一文件。fetchall程序完全忽略到了HTTP的響應Body隻是計算了一下響應Body的大小這個程序中把響應Body拷到了ioutil.Discard。在本的web服器程序中是用fmt.Fprintf直接到了http.ResponseWriter中。
些程序中,我看到了很多不同的型被出到标准输出流中。比如前面的fetch程序把HTTP的响应数据拷贝到了os.Stdoutlissajous程序里我们输出的是一文件。fetchall程序完全忽略到了HTTP的响应Body只是计算了一下响应Body的大小这个程序中把响应Body拷到了ioutil.Discard。在本的web服器程序中是用fmt.Fprintf直接到了http.ResponseWriter中。
管三種具體的實現流程不太一,他們都實現一個共同的接口,卽當它們被調用需要一個標準流輸出時都可以滿足。這個接口叫作io.Writer在7.1節中會詳細討論
管三种具体的实现流程不太一,他们都实现一个共同的接口,即当它们被调用需要一个标准流输出时都可以足。这个接口叫作io.Writer在7.1节中会详细讨论
Go言的接口機製會在第7章中解,了在這里簡單説明接口能做什麽,讓我們簡單地將這里的web服器和之前的lissajous函數結合起來,這樣GIF動畵可以被到HTTP的客端,而不是之前的標準輸出流。要在web服器的代里加入下面這幾行。
Go言的接口机制会在第7章中解,了在这里简单说明接口能做什么,让我们简单地将这里的web服器和之前的lissajous函数结合起来,这样GIF动画可以被到HTTP的客端,而不是之前的标准输出流。要在web服器的代里加入下面这几行。
```Go
handler := func(w http.ResponseWriter, r *http.Request) {
@@ -143,7 +143,7 @@ handler := func(w http.ResponseWriter, r *http.Request) {
http.HandleFunc("/", handler)
```
或者另一種等價形式:
或者另一种等价形式:
```Go
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
@@ -151,11 +151,11 @@ http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
})
```
HandleFunc函的第二個參數是一個函數的字面值,也就是一在使用時定義的匿名函數。這些內容我們會在5.6節中講解。
HandleFunc函的第二个参数是一个函数的字面值,也就是一在使用时定义的匿名函数。这些内容我们会在5.6节中讲解。
做完這些脩改之,在瀏覽器里訪問 http://localhost:8000 。每次你載入這個頁面都可以看到一個像圖1.3那樣的動畵
做完这些修改之,在浏览器里访问 http://localhost:8000 。每次你载入这个页面都可以看到一个像图1.3那样的动画
![](../images/ch1-03.png)
**練習 1.12** 改Lissajour服從URL讀取變量,比如你可以訪問 http://localhost:8000/?cycles=20 這個URL這樣訪問可以程序里的cycles默的5脩改爲20。字符串轉換爲數字可以調用strconv.Atoi函。你可以在godoc里看strconv.Atoi的詳細説明。
**练习 1.12** 改Lissajour服从URL读取变量,比如你可以访问 http://localhost:8000/?cycles=20 这个URL这样访问可以程序里的cycles默的5修改为20。字符串转换为数字可以用strconv.Atoi函。你可以在godoc里看strconv.Atoi的详细说明。