Golang websocket client读取数据

♀尐吖头ヾ 提交于 2019-12-02 05:07:04

Golang既可以写websocket的server端也可以写websocket的client端,前者网上的资料很多后者甚少,今天遇到写client的需求,在此做个总结。

  1. 建立连接。连接成功建立后,client和server均可以随时往数据通道里写数据同时也可以从中读取数据。
var wsurl = "wss://api.huobi.pro/ws"
var origin = "http://api.huobi.pro/"
ws, err := websocket.Dial(wsurl, "", origin)
if err != nil {
    panic(err)
}
  1. 写数据。在通道已建立的前提下,写数据操作通过一行代码即可完成:
func sendMessage(data []bytes) {
	ws.Write(msg)
}

3.1 读数据。最简单的方法是调用func (ws *Conn) Read(msg []byte) (n int, err error)方法,定义一个用来接收数据的[]byte数组当作参数传入,但是由于不知道server发来的数据长度,所以一般是定义一个足够大的字节[]byte数组,这样读取一来浪费内存二来处理起来麻烦,不建议使用;

func readMessage(ws *websocket.Conn) {
	data := make([]byte, 1024*10)
	_, err := ws.Read(data)
	if err != nil {
		log.Println(err)
	}
}

3.2 读数据。对于都是以json方式传输的数据,websocket包提供了将每条message读取到一个interface{}中的方法,等同于json.Unmarshal

func readJsonMessage(ws *websocket.Conn) {
	var data interfact{} // data的类型为接收的JSON类型struct
	err := websocket.Message.Receive(ws, data)
	if err != nil {
		log.Println(err)
	}
}

3.3 读数据。3.2是将接收到的数据直接unmarshal到struct里了,而我的需求比这个要麻烦一点:server发来的数据[]byte数组是压缩过的,所以接收到数据后第一步应该解压缩然后才能unmarshal,所以不能再用3.2的方式,参照3.2的源码,实现方式如下。

func readOriginMessage(ws *websocket.Conn) {
	again:
		fr, err := ws.NewFrameReader()
		if err != nil {
			log.Printf("new frame reader err %v", err)
			return
		}
		frame, err := ws.HandleFrame(fr)
		if err != nil {
			log.Printf("handle frame err %v", err)
			return
		}
		if frame == nil {
			goto again
		}
		
		bytes, err := ioutil.ReadAll(frame)
		if err != nil {
			log.Printf("read frame data err %v", err)
		}
		unzipData, err := utils.UnzipByte(bytes)
		if err != nil {
			log.Printf("unzip data err %v", err)
		}

		var message map[string]interface{}
		e := json.Unmarshal(unzipData, &message)
		if e != nil {
			log.Printf("unmarshal err %v", e)
		}
		
		log.Printf("message content= %+v", message)
}
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!