我尝试使用Apiary并制作了一个通用模板,将JSON发送到模拟服务器并获得此代码:
package main
import (
"encoding/json"
"fmt"
"github.com/jmcvetta/napping"
"log"
"net/http"
)
func main() {
url := "http://restapi3.apiary.io/notes"
fmt.Println("URL:>", url)
s := napping.Session{}
h := &http.Header{}
h.Set("X-Custom-Header", "myvalue")
s.Header = h
var jsonStr = []byte(`
{
"title": "Buy cheese and bread for breakfast."
}`)
var data map[string]json.RawMessage
err := json.Unmarshal(jsonStr, &data)
if err != nil {
fmt.Println(err)
}
resp, err := s.Post(url, &data, nil, nil)
if err != nil {
log.Fatal(err)
}
fmt.Println("response Status:", resp.Status())
fmt.Println("response Headers:", resp.HttpResponse().Header)
fmt.Println("response Body:", resp.RawText())
}
此代码不能正确发送JSON,但我不知道为什么。每次调用时JSON字符串都可以不同。我无法使用Struct
。
答案 0 :(得分:404)
我不熟悉午睡,但使用Golang的net/http
包工作正常(playground):
func main() {
url := "http://restapi3.apiary.io/notes"
fmt.Println("URL:>", url)
var jsonStr = []byte(`{"title":"Buy cheese and bread for breakfast."}`)
req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonStr))
req.Header.Set("X-Custom-Header", "myvalue")
req.Header.Set("Content-Type", "application/json")
client := &http.Client{}
resp, err := client.Do(req)
if err != nil {
panic(err)
}
defer resp.Body.Close()
fmt.Println("response Status:", resp.Status)
fmt.Println("response Headers:", resp.Header)
body, _ := ioutil.ReadAll(resp.Body)
fmt.Println("response Body:", string(body))
}
答案 1 :(得分:77)
您可以使用post
发布您的json。
values := map[string]string{"username": username, "password": password}
jsonValue, _ := json.Marshal(values)
resp, err := http.Post(authAuthenticatorUrl, "application/json", bytes.NewBuffer(jsonValue))
答案 2 :(得分:11)
除了标准的net / http包之外,你可以考虑使用我的GoRequest包裹net / http,让你的生活更轻松,而不必过多考虑json或struct。但是你也可以在一个请求中混合和匹配它们! (您可以在gorequest github页面中查看有关它的更多详细信息)
因此,最终您的代码将变为如下:
func main() {
url := "http://restapi3.apiary.io/notes"
fmt.Println("URL:>", url)
request := gorequest.New()
titleList := []string{"title1", "title2", "title3"}
for _, title := range titleList {
resp, body, errs := request.Post(url).
Set("X-Custom-Header", "myvalue").
Send(`{"title":"` + title + `"}`).
End()
if errs != nil {
fmt.Println(errs)
os.Exit(1)
}
fmt.Println("response Status:", resp.Status)
fmt.Println("response Headers:", resp.Header)
fmt.Println("response Body:", body)
}
}
这取决于您希望如何实现。我创建了这个库,因为我遇到了同样的问题,我希望代码更短,易于使用json,并且在代码库和生产系统中更易于维护。
答案 3 :(得分:3)
如果您已经有一个结构。
type Student struct {
Name string `json:"name"`
Address string `json:"address"`
}
// .....
body := &Student{
Name: "abc",
Address: "xyz",
}
buf := new(bytes.Buffer)
json.NewEncoder(buf).Encode(body)
req, _ := http.NewRequest("POST", url, buf)
res, e := client.Do(req)
if e != nil {
return e
}
defer res.Body.Close()
io.Copy(os.Stdout, res.Body)
答案 4 :(得分:2)
如 io.Pipe 中提到的,将 another answer 用于大型请求正文。这种方法通过将数据从 JSON 编码器流式传输到网络来避免在内存中构建整个请求正文。
这个答案建立在另一个答案的基础上,展示了如何处理错误。始终处理错误!
代码如下:
r, w := io.Pipe()
go func() {
w.CloseWithError(json.NewEncoder(w).Encode(data))
}()
// Ensure that read side of pipe is closed. This
// unblocks goroutine in scenario where http.Post
// errors out before reading the entire request body.
defer r.Close()
resp, err := http.Post(url, r)
if err != nil {
// Adjust error handling here to meet application requrirements.
log.Fatal(err)
}
defer resp.Body.Close()
// Use the response here.
答案 5 :(得分:1)
http 或 https 的 post 请求示例
def to_internal_value(self, data):
# here is the first place to manipulate data you can do what ever you want with
#your data
return super().to_internal_value(data)
答案 6 :(得分:0)
如果你有很多数据要发送,可以使用管道:
package main
import (
"encoding/json"
"io"
"net/http"
)
func main() {
m := map[string]int{"SNG_ID": 75498415}
r, w := io.Pipe()
go func() {
json.NewEncoder(w).Encode(m)
w.Close()
}()
http.Post("https://stackoverflow.com", "application/json", r)
}