前面我们分析了http包中如何根据请求的URL转发请求到对应的处理器处理的,现在我们深入理解处理器是如何解析、处理请求的。
主要内容
- Request结构
- 表单
- 文本处理
1.1 Request结构
Request结构是根据HTTP请求报文,并按实际情况定义的,除了HTTP请求报文中定义的概念外,还增加了Form字段等信息。以下是Request结构的定义:
type Request struct {
Method string // 请求的方法
URL *url.URL // 请求报文起始行中的URL,URL类型的指针
Proto string // "HTTP/1.0"
ProtoMajor int // 1
ProtoMinor int // 0
Header Header // 请求头部字段
Body io.ReadCloser // 请求主体
GetBody func() (io.ReadCloser, error)
ContentLength int64
TransferEncoding []string
Close bool
Host string
// 请求报文中的一些参数,包括表单字段等
Form url.Values
PostForm url.Values
MultipartForm *multipart.Form
Trailer Header
RemoteAddr string
RequestURI string
TLS *tls.ConnectionState
Cancel <-chan struct{}
Response *Response
ctx context.Context
}
1.2 URL结构
Request中有个比较重要的字段URL,表示请求行中的URL,定义如下:
type URL struct {
Scheme string // 方案
Opaque string //
User *Userinfo // 基本验证方式中的username和password信息
Host string // 主机字段
Path string // 路径
RawPath string //
ForceQuery bool //
RawQuery string // 查询字段
Fragment string // 分片字段
}
1.3 请求首部
请求和响应的首部都使用Header类型表示,header类型是一个映射(map)类型,表示HTTP首部中多个键值对。
type Header map[string][]string
1.4 请求主体
请求和响应的主体都由Request结构中的Body表示,这个字段是Reader和Closer接口的结合。
我们尝试一下对Request结构的使用:
package main
import (
"fmt"
"net/http"
)
func urlHandler(w http.ResponseWriter, r *http.Request) {
// Request中的URL结构
fmt.Fprintln(w, r.URL.Path)
}
func headerHandler(w http.ResponseWriter, r *http.Request) {
// Request结构中的Header是映射(map)类型,可以通过键值对进行操作,也可以使用map实现的Get, Set等方法操作。
fmt.Fprintln(w, r.Header)
fmt.Fprintln(w, r.Header.Get("Accept-Encoding"))
}
func bodyHandler(w http.ResponseWriter, r *http.Request) {
// Request结构中的body字段实现了Reader接口,所以可以使用Read方法。
len := r.ContentLength
body := make([]byte, len)
r.Body.Read(body)
fmt.Fprintln(w, string(body))
}
func main() {
http.HandleFunc("/url", urlHandler)
http.HandleFunc("/header", headerHandler)
http.HandleFunc("/body", bodyHandler)
err := http.ListenAndServe(":8000", nil)
if err != nil {
fmt.Println(err)
}
}
2.1 HTML表单
表单是客户端和服务端进行数据交互的载体。通常,表单由POST请求在请求体中传递的;当然也可以用GET请求传递,但是该表单数据将会在URL中也键值对的方式传递。
表单是一个包含表单元素的区域,表单元素(文本,下拉列表,单选框,文件等),Go中Request结构定义了Form字段,可以方便对表单进行操作。
<html>
<head>
<title></title>
</head>
<body>
<form action="/login" method="post">
用户名:<input type="text" name="username">
密码:<input type="password" name="password">
<input type="submit" value="登录">
</form>
</body>
</html>
package main
import (
"fmt"
"html/template"
"net/http"
)
func login(w http.ResponseWriter, r *http.Request) {
// 登录逻辑,GET方法时放回登录页面,POST解析用户、密码进行登录验证。
r.ParseForm() // 解析url传递的参数,对于POST请求则解析请求体(request body)
if r.Method == "POST" {
fmt.Println(r.Form["username"])
fmt.Println(r.Form["password"])
} else {
t, _ := template.ParseFiles("login.html")
t.Execute(w, nil)
}
}
func main() {
http.HandleFunc("/login", login)
err := http.ListenAndServe(":8000", nil)
if err != nil {
fmt.Println(err)
}
}
2.2 文件
Web用户经常会使用文件上传功能,比如上传头像图片等。我们来实现文件上传功能。
首先我们先了解一下form的enctype属性,它决定了请求时在发送键值对将会使用的格式。enctype的三种格式:
格式 | 说明 |
---|---|
application/x-www-form-urlencoded | form的默认值,使用此格式的时候,表单中的数据被编码成一个连续的“长查询字符串”,不同的键值对将使用&符号隔开,而键值之间使用等号(=)隔开。这种形式是跟URL编码一样的。如:username=admin&password=123 |
multipart/form-data | 将数据编码成MIME报文,表单中的每个键值对都带有各自的内容类型和内容配置(disposition) |
text/plain | 空格转换为 "+" 加号,但不对特殊字符编码。 |
multipart/form-data格式常用于文件上传功能,所以我们以这种格式来示范文件上传功能。
<html>
<head>
<title></title>
</head>
<body>
<form action="/upload" method="post" enctype="multipart/form-data">
<input type="file" name="uploadfile" />
<input type="submit" value="upload">
</form>
</body>
</html>
package main
import (
"fmt"
"html/template"
"io"
"net/http"
"os"
)
func upload(w http.ResponseWriter, r *http.Request) {
if r.Method == "POST" {
//
r.ParseMultipartForm(1024)
// 使用r.FormFile获取文件句柄,然后对文件进行存储等处理。
file, handler, err := r.FormFile("uploadfile")
defer file.Close()
if err != nil {
fmt.Println(err)
return
}
fmt.Fprintf(w, "%v", handler.Header)
// 假设已经有upload目录,存储文件。
f, err := os.OpenFile("../static/upload/"+handler.Filename, os.O_WRONLY|os.O_CREATE, 0666)
if err != nil {
fmt.Println(err)
return
}
defer f.Close()
io.Copy(f, file)
} else {
t, _ := template.ParseFiles("upload.html")
t.Execute(w, nil)
}
}
func main() {
http.HandleFunc("/upload", upload)
err := http.ListenAndServe(":8000", nil)
if err != nil {
fmt.Println(err)
}
}
本节详细说明了Go中的Request结构,以及结合表单的使用。接下来介绍cookie与session的使用。