go websocket

websocket介绍

The WebSocket Protocol  RFC6455,这个是WebSocket的RFC文档,所以内容非常全面(当然只涉及协议,不涉及具体实现),不过内容太多,如果是初次了解,可以挑自己感兴趣的看看。

《WebSocket 是什么原理?为什么可以实现持久连接?》,这个是知乎上的一篇文章,对于WebSocket讲的通俗易懂,可以了解一下。 

       在介绍WebSocket之前,我们先来了解一下为什么会出现WebSocket。我们知道HTTP协议(底层使用的是TCP协议)是无状态的,即一次Request,一个Response就结束了。实际中的场景就是客户端(比如浏览器)向服务器发送一次请求,然后服务器返回一个响应,一次交互就结束了,底层的TCP连接也会断掉,下次请求时,重新再创建新的连接。而且这种通信是被动式的,就是说服务器端不能主动向客户端发响应,只能是客户端一个Request,服务的一个Response这种模式(当然最新的协议里面,可能可以将多个Request合并一次发给服务端,但模型仍旧是这种模式)。

       如果你曾经使用TCP协议写过通信程序,应该非常熟悉那种模式:客户端和服务端(有时都没有清晰的界限)通过三步握手建立连接后,就可以相互随便发送数据,除非网络异常或者主动关闭,否则该TCP连接将一直存在。而WebSocket的出现就是为了在应用层实现类似传输层的TCP协议,当然它底层和HTTP一样使用的是TCP协议。这样我们就明白一些了,WebSocket不像HTTP协议,一次交互后就结束,它建立后会一直存在(除非主动断开或者网络异常等),而且客户端和服务端可以任意向对方发送数据,不再像以前那么“傻”了。也就是说,HTTP协议是一次性的,“单工的”;而WebSocket是真正意义上的长连接,且是全双工的。当然,上述提及的需求HTTP通过poll和轮循等方式也可以实现,但弊端非常多:

  • 服务器端需要在底层为每个HTTP连接维护一个TCP连接,比如一个用于发送消息,一个用于接收消息等。
  • 资源浪费,每次的HTTP请求中都需要携带消息头。
  • 客户端还必须通过一些手段知道哪些响应对应发出去的哪些请求。

 

go websocket

       Go官方的标准包里面提供了一个WebSocket的包 golang.org/x/net/websocket,但也说的很明确,这个包里面并没有实现WebSocket协议规定的一些特性,而推荐使用github.com/gorilla/websocket这个包。

代码示例,代码主要实现服务端向客户端写入数据,你可以通过channel Broadcast插入数据。

web页面的测试,可以在百度上选择一些在线的,如http://coolaf.com/tool/chattest

 

package controllers

import (
	"github.com/astaxie/beego"
	"github.com/gorilla/websocket"
	"net/http"
)

const (
	readBufferSize  = 1024
	writeBufferSize = 1024
)

type Client struct {
	conn     *websocket.Conn
	messages chan []byte
}

var (
	upgrader = websocket.Upgrader{
		ReadBufferSize:  readBufferSize,
		WriteBufferSize: writeBufferSize,
		CheckOrigin: func(r *http.Request) bool {
			return true
		},
	}
	clients    map[*Client]bool // 存储所有的链接
	addClients chan *Client     // 新链接进来的chan
	delClients chan *Client     // 删除退出的chan
	Broadcast  chan []byte      // 广播聊天的chan
)

func (c *Client) writePump() {
	defer func() {
		delClients <- c
		c.conn.Close()
	}()
	for {
		select {
		case message := <- c.messages:
			err := c.conn.WriteMessage(websocket.TextMessage, message)
			if err != nil {
				return
			}
		}
	}
}

func manager() {
	clients = make(map[*Client]bool)
	addClients = make(chan *Client)
	delClients = make(chan *Client)

	for {
		select {
		case message := <- Broadcast:
			for client := range clients {
				select {
				case client.messages <- message:
				default:
					close(client.messages)
					delete(clients, client)
				}
			}
		case client := <- addClients:
			clients[client] = true
		case client := <- delClients:
			if _, ok := clients[client]; ok {
				close(client.messages)
				delete(clients, client)
			}
		}
	}
}

type WebSocketController struct {
	beego.Controller
}

func (this *WebSocketController) Ws() {
	ws, err := upgrader.Upgrade(this.Ctx.ResponseWriter, this.Ctx.Request, nil)
	if err != nil {
		return
	}
	client := &Client{conn: ws, messages: make(chan []byte, 256)}
	addClients <- client
	go client.writePump()
}

func init() {
	Broadcast = make(chan []byte)
	go manager()
}

 

 

参考文章:

http://time-track.cn/websocket-and-golang.html

https://www.cnblogs.com/cornor/p/6178507.html

https://github.com/gorilla/websocket/tree/master/examples/chat

https://github.com/astaxie/build-web-application-with-golang/blob/master/zh/08.2.md

https://beego.me/docs/examples/chat.md

 

posted @ 2018-01-17 10:14  shhnwangjian  阅读(791)  评论(0编辑  收藏  举报