web服务端代码s.go:
package main
import (
"io"
"log"
"net/http"
)
func handlerHello(w http.ResponseWriter, r *http.Request) {
io.WriteString(w, "hello girls")
}
func main() {
http.HandleFunc("/hello", handlerHello) // 注册
err := http.ListenAndServe("localhost:8080", nil)
if err != nil {
log.Println(err)
}
}
go run s.go一下,跑起来, 然后在浏览器执行http://127.0.0.1:8080/hello (或者在命令行用curl发http请求也可以), 浏览器上的结果为:
hello girls
好简单。可以在客户端或者服务端抓包看下, 很典型的http req和rsp.
我们再来看一个有趣的问题, 修改s.go为:
package main
import (
"io"
"log"
"net/http"
)
func handlerHello(w http.ResponseWriter, r *http.Request) {
str := `
table border="1">
<tr>
<td>row 1, cell 1</td>
<td>row 1, cell 2</td>
</tr>
<tr>
<td>row 2, cell 1</td>
<td>row 2, cell 2</td>
</tr>
</table>
`
io.WriteString(w, str)
}
func main() {
http.HandleFunc("/hello", handlerHello) // 注册
err := http.ListenAndServe("localhost:8080", nil)
if err != nil {
log.Println(err)
}
}
再次重启服务并发请求, 浏览器上显示的内容是:
table border="1">
<tr>
<td>row 1, cell 1</td>
<td>row 1, cell 2</td>
</tr>
<tr>
<td>row 2, cell 1</td>
<td>row 2, cell 2</td>
</tr>
</table>
抓包看一下, 发现有:Content-Type: text/plain; charset=utf-8
因此, 浏览器需要根据纯文本显示。 注意到, 上述的table左边少了一个"<". 我们加上后, s.go的代码如下:
package main
import (
"io"
"log"
"net/http"
)
func handlerHello(w http.ResponseWriter, r *http.Request) {
str := `
<table border="1">
<tr>
<td>row 1, cell 1</td>
<td>row 1, cell 2</td>
</tr>
<tr>
<td>row 2, cell 1</td>
<td>row 2, cell 2</td>
</tr>
</table>
`
io.WriteString(w, str)
}
func main() {
http.HandleFunc("/hello", handlerHello) // 注册
err := http.ListenAndServe("localhost:8080", nil)
if err != nil {
log.Println(err)
}
}
再次重启服务,发请求,浏览器端的显示是:
row 1, cell 1 | row 1, cell 2 |
row 2, cell 1 | row 2, cell 2 |
抓包看, 有Content-Type: text/html; charset=utf-8
可见, 服务端会判断str的格式,来确定Content-Type的类型, 从而决定了浏览器端的展示方式。服务端的自动判断行为, 有点意思。 在我看来, 这样不太好,应该让程序员来指定Content-Type.
不多说。