小伙伴们有没有觉得学习Golang很有意思?有意思就对了!今天就给大家带来《如何从gin中的前端获取数据?》,以下内容将会涉及到,若是在学习中对其中部分知识点有疑问,或许看了本文就能帮到你!
问题内容令我羞愧的是,我一直无法弄清楚如何从 gin 框架的前端获取数据。在 django 中我获取数据所以:
user=request.data.get('user')
print(user)
一切都简单易懂。 我应该如何在杜松子酒中做到这一点?
user := c.Query("user")
user := c.Param("user")
user := c.Params.ByName("user")
user := c.PostForm("user")
println(user)//emptiness....
解决方案
服务器 gin 无法处理来自 axios 的常规默认 application/json 请求!!!什么??? 请求应作为 application/x-www-form-urlencoded 发送。 我在vue项目中的决定: 使用 vue-resource 代替 axios (axios.post=>this.$http.post) 并带有选项 vue.http.options.emulatejson = true;在 main.js 中
嗯,我想说你应该找一些关于 http 如何工作的书/howto 并花一些时间来研究它,因为看起来你试图在没有真正理解的情况下解决手头的问题您的浏览器和后端服务之间会发生什么。
这里真正的问题是,您似乎意识到了更多的移动部分,而要走的路取决于您的前端所做的事情。
您没有准确地告诉我们您如何执行您的请求, 但从征求的评论看来,你正在使用那个“axios”。 如果我能正确地用谷歌搜索该项目, 其自述文件 states:
这意味着两件事:
- 除非你以某种方式调整了 axios 的设置,否则当你这样做时
axios.post
,据说是执行 http post 请求 其content-type
字段设置为application/json
及其有效负载(或“主体”,如果您愿意的话)是{user:this.user}
javascript 对象的 json 序列化。 - 因此尝试解析查询字符串是徒劳的。 尝试将请求解析为 http 表单是徒劳的——事实并非如此。
相反,您应该将传入请求的正文解释为 json 格式。我不知道如何在“go-gin”中做到这一点,但在普通的 go 中,这会是这样的
func (h *handler) servehttp(rw http.responsewriter, req *http.request) {
defer req.body.close()
var user user
dec := json.newdecoder(req.body)
err := dec.decode(&user)
if err != nil {
rw.header().set("content-type", "text/plain; charset=utf-8")
rw.writeheader(http.statusbadrequest)
fmt.fprintln(rw, "error parsing request body: ", err)
return
}
}
理想情况下,您首先检查传入请求的内容类型是否确实是 application/json
,如果不是,则立即使用 http.statusbadrequest
拒绝它。
执行此操作的工作代码示例是
// verifycontenttypeisjson makes sure the http header of a server
// http.request contains the content-type field and it indicates
// the request payload is json.
// the implementation is based on rfc 7231 (section 3.1.1.5) and rfc 8259.
func verifycontenttypeisjson(header http.header) error {
var s string
if values := header["content-type"]; len(values) > 0 {
s = values[0]
} else {
return errors.new("missing content-type")
}
if s == "" {
return errors.new("empty content-type")
}
if i := strings.indexbyte(s, ';'); i != -1 {
s = strings.trimspace(s[:i])
}
if strings.tolower(s) != "application/json" {
return fmt.errorf("unknown content-type: %v, must be application/json", s)
}
return nil
}
有了这个函数,你就会得到这样的东西
在 defer req.body.close()
并实际解析它之后:
if err := verifycontenttypeisjson(req.header); err != nil {
rw.header().set("content-type", "text/plain; charset=utf-8")
rw.writeheader(http.statusbadrequest)
fmt.fprintln(rw, err)
return
}
(请注意,“go-gin”可能已经内置了类似于此的内容,因此请检查这一点。)
user
类型应该是某种 struct
类型,与您打算从请求中解组的 json 对象的形状相匹配。像这样的事情:
type User struct {
User string `json:"user"`
}
没有,在这两个地方我的示例返回了 使用纯文本内容类型向用户发出错误 (采用 utf-8 编码)。这可能没问题,但也可能不好。 比如说,您的客户可能需要 json 格式的文档 一些商定的形状。
或者您可以使用 content negotiation,但我建议先把简单的事情搞清楚。
要检查的文献:
http post
请求 explained at MDN。- URL's query string。
- xhr explained at MDN — 另请参阅此处的链接。
- "Writing Web Applications in Go", 一般情况下为 this。
或许可以回答你的问题的一部分 为什么它在 django 中“正常工作”。 我只能猜测,但我认为它只是实现了大量的魔法,它查看传入的请求并尝试猜测如何从中提取数据。
问题是猜测确实可能适用于 一次性脚本,但是当你要实现像 web api 这样的东西时(许多人不太正确地称之为“rest”,但我们不要离题),这是最好的 非常明确地说明您的端点接受什么 准确以及他们如何准确响应请求 - 合法的和非格式良好的请求。
关于go中的魔法,你可以阅读this。
今天带大家了解了的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注编程网公众号,一起学习编程~