• golang从0到1实战系统四十:处理表单的输入


    先来看一个表单递交的例子,我们有如下的表单内容,命名成文件 login.gtpl(放入当前新建
    项目的目录里面)

    <html>
    <head>
    <title></title>
    </head>
    <body>
    <form action="http://127.0.0.1:9090/login" method="post">
     用户名:<input type="text" name="username">
     密码:<input type="password" name="password">
     <input type="submit" value="登陆">
    </form>
    </body>
    </html>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    上面递交表单到服务器/login,当用户输入信息点击登陆之后,会跳转到服务器的路由
    login 里面,我们首先要判断这个是什么方式传递过来,POST 还是 GET 呢?

    http 包里面有一个很简单的方式就可以获取,我们在前面 web 的例子的基础上来看看怎么
    处理 login 页面的 form 数据

    package main
    import (
     "fmt"
     "html/template"
     "log"
     "net/http"
     "strings"
    )
    func sayhelloName(w http.ResponseWriter, r *http.Request) {
     r.ParseForm() //解析 url 传递的参数,对于 POST 则解析响应包的主体(request body)
     //注意:如果没有调用 ParseForm 方法,下面无法获取表单的数据
     fmt.Println(r.Form) //这些信息是输出到服务器端的打印信息
     fmt.Println("path", r.URL.Path)
     fmt.Println("scheme", r.URL.Scheme)
     fmt.Println(r.Form["url_long"])
     for k, v := range r.Form {
     fmt.Println("key:", k)
     fmt.Println("val:", strings.Join(v, ""))
     }
     fmt.Fprintf(w, "Hello astaxie!") //这个写入到 w 的是输出到客户端的
    }
    func login(w http.ResponseWriter, r *http.Request) {
     fmt.Println("method:", r.Method) //获取请求的方法
     if r.Method == "GET" {
     t, _ := template.ParseFiles("login.gtpl")
     t.Execute(w, nil)
     } else {
     //请求的是登陆数据,那么执行登陆的逻辑判断
     fmt.Println("username:", r.Form["username"])
     fmt.Println("password:", r.Form["password"])
     }
    }
    func main() {
     http.HandleFunc("/", sayhelloName) //设置访问的路由
     http.HandleFunc("/login", login) //设置访问的路由
     err := http.ListenAndServe(":9090", nil) //设置监听的端口
     if err != nil {
     log.Fatal("ListenAndServe: ", err)
     }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40

    通过上面的代码我们可以看出获取请求方法是通过 r.Method 来完成的,这是个字符串类型
    的变量,返回 GET, POST, PUTmethod 信息。

    login 函数中我们根据 r.Method 来判断是显示登录界面还是处理登录逻辑。当 GET 方式请求
    时显示登录界面,其他方式请求时则处理登录逻辑,如查询数据库、验证登录信息等。
    当我们在浏览器里面打开 http://127.0.0.1:9090/login 的时候,出现如下界面

    我们输入用户名和密码之后发现在服务器端是不会打印出来任何输出的,为什么呢?

    默认情况下,Handler 里面是不会自动解析 form 的,必须显式的调用 r.ParseForm()后,你才能
    对这个表单数据进行操作。我们修改一下代码,在 fmt.Println("username:", r.Form["username"])之前加一行 r.ParseForm(),重新编译,再次测试输入递交,现在是不是在服务器端有输出你的输入的用户名和密码了。

    r.Form 里面包含了所有请求的参数,比如 URL 中 query-string、POST 的数据、PUT 的数据,
    所有当你在 URL 的 query-string 字段和 POST 冲突时,会保存成一个 slice,里面存储了多
    个值,Go 官方文档中说在接下来的版本里面将会把 POST、GET 这些数据分离开来。

    现在我们修改一下 login.gtpl 里面 formactionhttp://127.0.0.1:9090/login 修改为
    http://127.0.0.1:9090/login?username=astaxie,再次测试,服务器的输出 username 是不
    是一个 slice。服务器端的输出如下:

    request.Form 是一个 url.Values 类型,里面存储的是对应的类似 key=value 的信息,下面
    展示了可以对 form 数据进行的一些操作:

    v := url.Values{}
    v.Set("name", "Ava")
    v.Add("friend", "Jess")
    v.Add("friend", "Sarah")
    v.Add("friend", "Zoe")
    // v.Encode() == 
    "name=Ava&friend=Jess&friend=Sarah&friend=Zoe"
    fmt.Println(v.Get("name"))
    fmt.Println(v.Get("friend"))
    fmt.Println(v["friend"])
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    Tips: Request 本身也提供了 FormValue()函数来获取用户提交的参数。如
    r.Form[“username”]也可写成 r.FormValue(“username”)。调用 r.FormValue 时会自动调用
    r.ParseForm,所以不必提前调用。r.FormValue 只会返回同名参数中的第一个,若参数不存
    在则返回空字符串。

  • 相关阅读:
    嵌入式软件架构设计-建立抽象层
    SystemVerilog Assertions应用指南 第一章(1.28章节 内建的系统函数)
    员工脉动调查中十个最佳问题
    php实现普通和定时跳转的几种方式
    接口幂等性
    23种设计模式(十三)组合模式(阁瑞钛伦特软件-九耶实训)
    HttpServletRequest相关
    C++命名空间
    《MongoDB入门教程》第01篇 MongoDB简介
    大规模 Spring Cloud 微服务无损上下线探索与实践
  • 原文地址:https://blog.csdn.net/Mr_Roki/article/details/136447487