|
8 | 8 | 你想要确保从一个表单元素中得到一个值,例如前面小节里面的用户名,我们如何处理呢?Go有一个内置函数`len`可以获取字符串的长度,这样我们就可以通过len来获取数据的长度,例如:
|
9 | 9 | ```Go
|
10 | 10 |
|
11 |
| - if len(r.Form["username"][0])==0{ |
12 |
| - //为空的处理 |
13 |
| - } |
| 11 | +if len(r.Form["username"][0])==0{ |
| 12 | + //为空的处理 |
| 13 | +} |
14 | 14 | ```
|
15 | 15 | `r.Form`对不同类型的表单元素的留空有不同的处理, 对于空文本框、空文本区域以及文件上传,元素的值为空值,而如果是未选中的复选框和单选按钮,则根本不会在r.Form中产生相应条目,如果我们用上面例子中的方式去获取数据时程序就会报错。所以我们需要通过`r.Form.Get()`来获取值,因为如果字段不存在,通过该方式获取的是空值。但是通过`r.Form.Get()`只能获取单个的值,如果是map的值,必须通过上面的方式来获取。
|
16 | 16 |
|
|
20 | 20 | 如果我们是判断正整数,那么我们先转化成int类型,然后进行处理
|
21 | 21 | ```Go
|
22 | 22 |
|
23 |
| - getint,err:=strconv.Atoi(r.Form.Get("age")) |
24 |
| - if err!=nil{ |
25 |
| - //数字转化出错了,那么可能就不是数字 |
26 |
| - } |
| 23 | +getint,err:=strconv.Atoi(r.Form.Get("age")) |
| 24 | +if err!=nil{ |
| 25 | + //数字转化出错了,那么可能就不是数字 |
| 26 | +} |
27 | 27 |
|
28 |
| - //接下来就可以判断这个数字的大小范围了 |
29 |
| - if getint >100 { |
30 |
| - //太大了 |
31 |
| - } |
| 28 | +//接下来就可以判断这个数字的大小范围了 |
| 29 | +if getint >100 { |
| 30 | + //太大了 |
| 31 | +} |
32 | 32 | ```
|
33 | 33 | 还有一种方式就是正则匹配的方式
|
34 | 34 | ```Go
|
35 | 35 |
|
36 |
| - if m, _ := regexp.MatchString("^[0-9]+$", r.Form.Get("age")); !m { |
37 |
| - return false |
38 |
| - } |
| 36 | +if m, _ := regexp.MatchString("^[0-9]+$", r.Form.Get("age")); !m { |
| 37 | + return false |
| 38 | +} |
39 | 39 | ```
|
40 | 40 | 对于性能要求很高的用户来说,这是一个老生常谈的问题了,他们认为应该尽量避免使用正则表达式,因为使用正则表达式的速度会比较慢。但是在目前机器性能那么强劲的情况下,对于这种简单的正则表达式效率和类型转换函数是没有什么差别的。如果你对正则表达式很熟悉,而且你在其它语言中也在使用它,那么在Go里面使用正则表达式将是一个便利的方式。
|
41 | 41 |
|
|
45 | 45 | 有时候我们想通过表单元素获取一个用户的中文名字,但是又为了保证获取的是正确的中文,我们需要进行验证,而不是用户随便的一些输入。对于中文我们目前有两种方式来验证,可以使用 `unicode` 包提供的 `func Is(rangeTab *RangeTable, r rune) bool` 来验证,也可以使用正则方式来验证,这里使用最简单的正则方式,如下代码所示
|
46 | 46 | ```Go
|
47 | 47 |
|
48 |
| - if m, _ := regexp.MatchString("^\\p{Han}+$", r.Form.Get("realname")); !m { |
49 |
| - return false |
50 |
| - } |
| 48 | +if m, _ := regexp.MatchString("^\\p{Han}+$", r.Form.Get("realname")); !m { |
| 49 | + return false |
| 50 | +} |
51 | 51 | ```
|
52 | 52 | ## 英文
|
53 | 53 | 我们期望通过表单元素获取一个英文值,例如我们想知道一个用户的英文名,应该是astaxie,而不是asta谢。
|
54 | 54 |
|
55 | 55 | 我们可以很简单的通过正则验证数据:
|
56 | 56 | ```Go
|
57 | 57 |
|
58 |
| - if m, _ := regexp.MatchString("^[a-zA-Z]+$", r.Form.Get("engname")); !m { |
59 |
| - return false |
60 |
| - } |
| 58 | +if m, _ := regexp.MatchString("^[a-zA-Z]+$", r.Form.Get("engname")); !m { |
| 59 | + return false |
| 60 | +} |
61 | 61 |
|
62 | 62 | ```
|
63 | 63 | ## 电子邮件地址
|
64 | 64 | 你想知道用户输入的一个Email地址是否正确,通过如下这个方式可以验证:
|
65 | 65 | ```Go
|
66 | 66 |
|
67 |
| - if m, _ := regexp.MatchString(`^([\w\.\_]{2,10})@(\w{1,}).([a-z]{2,4})$`, r.Form.Get("email")); !m { |
68 |
| - fmt.Println("no") |
69 |
| - }else{ |
70 |
| - fmt.Println("yes") |
71 |
| - } |
| 67 | +if m, _ := regexp.MatchString(`^([\w\.\_]{2,10})@(\w{1,}).([a-z]{2,4})$`, r.Form.Get("email")); !m { |
| 68 | + fmt.Println("no") |
| 69 | +}else{ |
| 70 | + fmt.Println("yes") |
| 71 | +} |
72 | 72 |
|
73 | 73 | ```
|
74 | 74 | ## 手机号码
|
75 | 75 | 你想要判断用户输入的手机号码是否正确,通过正则也可以验证:
|
76 | 76 | ```Go
|
77 | 77 |
|
78 |
| - if m, _ := regexp.MatchString(`^(1[3|4|5|8][0-9]\d{4,8})$`, r.Form.Get("mobile")); !m { |
79 |
| - return false |
80 |
| - } |
| 78 | +if m, _ := regexp.MatchString(`^(1[3|4|5|8][0-9]\d{4,8})$`, r.Form.Get("mobile")); !m { |
| 79 | + return false |
| 80 | +} |
81 | 81 | ```
|
82 | 82 | ## 下拉菜单
|
83 | 83 | 如果我们想要判断表单里面`<select>`元素生成的下拉菜单中是否有被选中的项目。有些时候黑客可能会伪造这个下拉菜单不存在的值发送给你,那么如何判断这个值是否是我们预设的值呢?
|
84 | 84 |
|
85 | 85 | 我们的select可能是这样的一些元素
|
86 | 86 | ```html
|
87 | 87 |
|
88 |
| - <select name="fruit"> |
89 |
| - <option value="apple">apple</option> |
90 |
| - <option value="pear">pear</option> |
91 |
| - <option value="banane">banane</option> |
92 |
| - </select> |
| 88 | +<select name="fruit"> |
| 89 | +<option value="apple">apple</option> |
| 90 | +<option value="pear">pear</option> |
| 91 | +<option value="banane">banane</option> |
| 92 | +</select> |
93 | 93 | ```
|
94 | 94 | 那么我们可以这样来验证
|
95 | 95 | ```Go
|
96 | 96 |
|
97 |
| - slice:=[]string{"apple","pear","banane"} |
98 |
| - |
99 |
| - v := r.Form.Get("fruit") |
100 |
| - for _, item range slice { |
101 |
| - if item == v { |
102 |
| - return true |
103 |
| - } |
| 97 | +slice:=[]string{"apple","pear","banane"} |
| 98 | + |
| 99 | +v := r.Form.Get("fruit") |
| 100 | +for _, item range slice { |
| 101 | + if item == v { |
| 102 | + return true |
104 | 103 | }
|
105 |
| - |
106 |
| - return false |
| 104 | +} |
| 105 | + |
| 106 | +return false |
107 | 107 | ```
|
108 | 108 | ## 单选按钮
|
109 | 109 | 如果我们想要判断radio按钮是否有一个被选中了,我们页面的输出可能就是一个男、女性别的选择,但是也可能一个15岁大的无聊小孩,一手拿着http协议的书,另一只手通过telnet客户端向你的程序在发送请求呢,你设定的性别男值是1,女是2,他给你发送一个3,你的程序会出现异常吗?因此我们也需要像下拉菜单的判断方式类似,判断我们获取的值是我们预设的值,而不是额外的值。
|
110 | 110 | ```html
|
111 | 111 |
|
112 |
| - <input type="radio" name="gender" value="1">男 |
113 |
| - <input type="radio" name="gender" value="2">女 |
| 112 | +<input type="radio" name="gender" value="1">男 |
| 113 | +<input type="radio" name="gender" value="2">女 |
114 | 114 | ```
|
115 | 115 | 那我们也可以类似下拉菜单的做法一样
|
116 | 116 | ```Go
|
117 | 117 |
|
118 |
| - slice:=[]int{1,2} |
| 118 | +slice:=[]int{1,2} |
119 | 119 |
|
120 |
| - for _, v := range slice { |
121 |
| - if v == r.Form.Get("gender") { |
122 |
| - return true |
123 |
| - } |
| 120 | +for _, v := range slice { |
| 121 | + if v == r.Form.Get("gender") { |
| 122 | + return true |
124 | 123 | }
|
125 |
| - return false |
| 124 | +} |
| 125 | +return false |
126 | 126 | ```
|
127 | 127 | ## 复选框
|
128 | 128 | 有一项选择兴趣的复选框,你想确定用户选中的和你提供给用户选择的是同一个类型的数据。
|
129 | 129 | ```html
|
130 | 130 |
|
131 |
| - <input type="checkbox" name="interest" value="football">足球 |
132 |
| - <input type="checkbox" name="interest" value="basketball">篮球 |
133 |
| - <input type="checkbox" name="interest" value="tennis">网球 |
| 131 | +<input type="checkbox" name="interest" value="football">足球 |
| 132 | +<input type="checkbox" name="interest" value="basketball">篮球 |
| 133 | +<input type="checkbox" name="interest" value="tennis">网球 |
134 | 134 | ```
|
135 | 135 | 对于复选框我们的验证和单选有点不一样,因为接收到的数据是一个slice
|
136 | 136 | ```Go
|
137 | 137 |
|
138 |
| - slice:=[]string{"football","basketball","tennis"} |
139 |
| - a:=Slice_diff(r.Form["interest"],slice) |
140 |
| - if a == nil{ |
141 |
| - return true |
142 |
| - } |
| 138 | +slice:=[]string{"football","basketball","tennis"} |
| 139 | +a:=Slice_diff(r.Form["interest"],slice) |
| 140 | +if a == nil{ |
| 141 | + return true |
| 142 | +} |
143 | 143 |
|
144 |
| - return false |
| 144 | +return false |
145 | 145 | ```
|
146 | 146 | 上面这个函数`Slice_diff`包含在我开源的一个库里面(操作slice和map的库),[https://github.com/astaxie/beeku](https://github.com/astaxie/beeku)
|
147 | 147 |
|
|
152 | 152 | Go里面提供了一个time的处理包,我们可以把用户的输入年月日转化成相应的时间,然后进行逻辑判断
|
153 | 153 | ```Go
|
154 | 154 |
|
155 |
| - t := time.Date(2009, time.November, 10, 23, 0, 0, 0, time.UTC) |
156 |
| - fmt.Printf("Go launched at %s\n", t.Local()) |
| 155 | +t := time.Date(2009, time.November, 10, 23, 0, 0, 0, time.UTC) |
| 156 | +fmt.Printf("Go launched at %s\n", t.Local()) |
157 | 157 | ```
|
158 | 158 | 获取time之后我们就可以进行很多时间函数的操作。具体的判断就根据自己的需求调整。
|
159 | 159 |
|
160 | 160 | ## 身份证号码
|
161 | 161 | 如果我们想验证表单输入的是否是身份证,通过正则也可以方便的验证,但是身份证有15位和18位,我们两个都需要验证
|
162 | 162 | ```Go
|
163 | 163 |
|
164 |
| - //验证15位身份证,15位的是全部数字 |
165 |
| - if m, _ := regexp.MatchString(`^(\d{15})$`, r.Form.Get("usercard")); !m { |
166 |
| - return false |
167 |
| - } |
| 164 | +//验证15位身份证,15位的是全部数字 |
| 165 | +if m, _ := regexp.MatchString(`^(\d{15})$`, r.Form.Get("usercard")); !m { |
| 166 | + return false |
| 167 | +} |
| 168 | + |
| 169 | +//验证18位身份证,18位前17位为数字,最后一位是校验位,可能为数字或字符X。 |
| 170 | +if m, _ := regexp.MatchString(`^(\d{17})([0-9]|X)$`, r.Form.Get("usercard")); !m { |
| 171 | + return false |
| 172 | +} |
168 | 173 |
|
169 |
| - //验证18位身份证,18位前17位为数字,最后一位是校验位,可能为数字或字符X。 |
170 |
| - if m, _ := regexp.MatchString(`^(\d{17})([0-9]|X)$`, r.Form.Get("usercard")); !m { |
171 |
| - return false |
172 |
| - } |
173 |
| - |
174 | 174 | ```
|
175 | 175 | 上面列出了我们一些常用的服务器端的表单元素验证,希望通过这个引导入门,能够让你对Go的数据验证有所了解,特别是Go里面的正则处理。
|
176 | 176 |
|
|
0 commit comments