用go发送JSON

时间:2013-05-19 12:19:12

标签: json http go

我正在尝试使用Go发送JSON消息。 这是服务器代码:

func (network *Network) Join(
    w http.ResponseWriter,
    r *http.Request) {
    //the request is not interesting
    //the response will be a message with just the clientId value set
    log.Println("client wants to join")
    message := Message{-1, -1, -1, ClientId(len(network.Clients)), -1, -1}
    var buffer bytes.Buffer
    enc := json.NewEncoder(&buffer)

    err := enc.Encode(message)
    if err != nil {
        fmt.Println("error encoding the response to a join request")
        log.Fatal(err)
    }

    fmt.Printf("the json: %s\n", buffer.Bytes())
    fmt.Fprint(w, buffer.Bytes())
}

网络是一种自定义结构。在main函数中,我正在创建一个网络对象并将其方法注册为http.HandleFunc(...)的回调

func main() {
    runtime.GOMAXPROCS(2)
    var network = new(Network)
    var clients = make([]Client, 0, 10)
    network.Clients = clients

    log.Println("starting the server")
    http.HandleFunc("/request", network.Request)
    http.HandleFunc("/update", network.GetNews)
    http.HandleFunc("/join", network.Join)
    log.Fatal(http.ListenAndServe("localhost:5000", nil))
}

Message也是一个结构。它有六个字段,都是int的类型别名。 当客户端向URL“localhost:5000 / join”发送http GET请求时,应该会发生这种情况

  • 方法是在网络对象上加入
  • 创建了一个具有客户端ID的新Message对象
  • 此邮件编码为JSON
  • 要检查编码是否正确,编码的消息将打印在cmd
  • 邮件将写入ResponseWriter

客户很简单。它具有与Message结构完全相同的代码。在main函数中,它只是向“localhost:5000 / join”发送一个GET请求,并尝试解码响应。这是代码

func main() {

    // try to join
    var clientId ClientId
    start := time.Now()
    var message Message
    resp, err := http.Get("http://localhost:5000/join")
    if err != nil {
        log.Fatal(err)
    }

    fmt.Println(resp.Status)
    dec := json.NewDecoder(resp.Body)
    err = dec.Decode(&message)
    if err != nil {
        fmt.Println("error decoding the response to the join request")
        log.Fatal(err)
    }

    fmt.Println(message)
    duration := time.Since(start)
    fmt.Println("connected after: ", duration)
    fmt.Println("with clientId", message.ClientId)
}

我启动了服务器,等了几秒钟然后运行了客户端。这是结果

  • 服务器打印“客户想要加入”
  • 服务器打印“json:{”What“: - 1,”Tag“: - 1,”Id“: - 1,”ClientId“:0,”X“: - 1,”Y“: - 1}“
  • 客户打印“200 OK”
  • 客户端崩溃“解码对加入请求的响应的错误”
  • 错误是“无效字符”3“数组元素后”

此错误消息让我很困惑。毕竟,在我的json中没有任何地方,那就是数字3.所以我在客户端上导入了io / ioutil,只是用这段代码打印了响应

b, _ := ioutil.ReadAll(resp.Body)
fmt.Printf("the json: %s\n", b)

请注意,print语句与服务器上的相同。我希望看到我编码的JSON。相反,我得到了这个

  • “200 OK”
  • “json:[123 34 87 104 97 116 ....]”这个名单已经持续了很长时间

我是新来的,不知道我是否正确地做到了这一点。但似乎上面的代码只是打印了一些字节。奇怪的是,在服务器上输出转换为字符串。

我的猜测是,我正在读取错误的数据,或者在服务器和客户端之间的路上损坏了消息。但老实说,这些只是猜测。

2 个答案:

答案 0 :(得分:4)

在您的服务器中,而不是

fmt.Fprint(w, buffer.Bytes())

你需要使用:

w.Write(buffer.Bytes())

fmt包将Bytes()格式化为人类可读的切片,其字节表示为整数,如下所示:

[123 34 87 104 97 116 ... etc

答案 1 :(得分:3)

您不想使用fmt.Print将内容写入响应。例如

package main

import (
    "fmt"
    "os"
)

func main() {
    bs := []byte("Hello, playground")
    fmt.Fprint(os.Stdout, bs)
}

playground link

可生产

[72 101 108 108 111 44 32 112 108 97 121 103 114 111 117 110 100]

使用Write() method of the ResponseWriter代替

您可以通过telneting到您的服务器作为实验找到这一点 - 当您不确定发生了什么时,总是一个好主意!