golang的websocket封装

栏目: Html5 · 发布时间: 7年前

内容简介:实现代码如下:首先定义一个结构体wsConn websocket的长链接的实体

实现代码如下:

import (
    "errors"
    "github.com/gorilla/websocket"
    "sync"
)

type Connection struct {
    wsConn *websocket.Conn
    inChan chan []byte
    outChan chan []byte
    closeChan chan byte
    mutex sync.Mutex
    isClosed   bool
}

func InitConnection(wsConn *websocket.Conn)(conn *Connection ,err error){
    conn=&Connection{
        wsConn:wsConn,
        inChan:make(chan []byte,1000),
        outChan:make(chan []byte,1000),
        closeChan:make(chan byte,1),
    }
    go conn.readLoop()
    go conn.writeLoop()
    return
}

func (conn *Connection) ReadMessage()(data []byte,err error){
    select {
    case data=<-conn.inChan:
    case <-conn.closeChan:
        err=errors.New("connection is closed")
    }
    return
}

func (conn *Connection) WriteMessage(data []byte)(err error){
        select {
        case conn.outChan<-data:
        case <-conn.closeChan:
            err=errors.New("connection is closed")
        }
        return
}

func (conn *Connection) Close(){
    //线程安全的Close,可重入
    conn.wsConn.Close()
    conn.mutex.Lock()
    if !conn.isClosed {
        close(conn.closeChan)
        conn.isClosed=true
    }
    conn.mutex.Unlock()

}

func (conn *Connection) readLoop(){
    var(
        data []byte
        err error
    )
    for{
        if _,data,err=conn.wsConn.ReadMessage();err !=nil{
            goto ERR
        }
        select {
        case conn.inChan<-data:
        case <-conn.closeChan:
            goto ERR
        }

    }
    ERR:
        conn.Close()
}
func (conn *Connection) writeLoop(){
    var (
        data []byte
        err error
    )
    for{
        select {
        case data=<-conn.outChan:
        case <-conn.closeChan:
            goto ERR
        }
        if err=conn.wsConn.WriteMessage(websocket.TextMessage,data);err!=nil{
            goto ERR
        }
    }
    ERR:
        conn.Close()
}

首先定义一个结构体

type Connection struct {
    wsConn *websocket.Conn
    inChan chan []byte
    outChan chan []byte
    closeChan chan byte
    mutex sync.Mutex
    isClosed   bool
}

wsConn websocket的长链接的实体

inChan 读数据的channel

outChan 写数据的channel

closeChan 链接关闭的channel

mutex 互斥锁

isClosed 链接关闭标识符

func InitConnection(wsConn *websocket.Conn)(conn *Connection ,err error){
    conn=&Connection{
        wsConn:wsConn,
        inChan:make(chan []byte,1000),
        outChan:make(chan []byte,1000),
        closeChan:make(chan byte,1),
    }
    go conn.readLoop()
    go conn.writeLoop()
    return
}

初始化链接

readLoop 和writeLoop 循环从websocket中读取数据和写入数据

func (conn *Connection) ReadMessage()(data []byte,err error){
    select {
    case data=<-conn.inChan:
    case <-conn.closeChan:
        err=errors.New("connection is closed")
    }
    return
}

ReadMessage() 从inChan中读取数据

func (conn *Connection) WriteMessage(data []byte)(err error){
        select {
        case conn.outChan<-data:
        case <-conn.closeChan:
            err=errors.New("connection is closed")
        }
        return
}

WriteMessage(data []byte) 写入数据传递给outChan ,writeLoop 监听outChan并写入数据

func (conn *Connection) Close(){
    //线程安全的Close,可重入
    conn.wsConn.Close()
    conn.mutex.Lock()
    if !conn.isClosed {
        close(conn.closeChan)
        conn.isClosed=true
    }
    conn.mutex.Unlock()
}

mutex锁住关闭操作 ,避免重复循环关闭链接

关闭链接时,传递closeChan ,同时关闭readLoop 和writeLoop

func (conn *Connection) readLoop(){
    var(
        data []byte
        err error
    )
    for{
        if _,data,err=conn.wsConn.ReadMessage();err !=nil{
            goto ERR
        }
        select {
        case conn.inChan<-data:
        case <-conn.closeChan:
            goto ERR
        }

    }
    ERR:
        conn.Close()
}

func (conn *Connection) writeLoop(){
    var (
        data []byte
        err error
    )
    for{
        select {
        case data=<-conn.outChan:
        case <-conn.closeChan:
            goto ERR
        }
        if err=conn.wsConn.WriteMessage(websocket.TextMessage,data);err!=nil{
            goto ERR
        }
    }
    ERR:
        conn.Close()
}

通过outChan和inChan 传递信息,保证线程的安全。


以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

Node.js:来一打 C++ 扩展

Node.js:来一打 C++ 扩展

死月 / 电子工业出版社 / 2018-6-1 / 109

Node.js 作为近几年新兴的一种编程运行时,托 V8 引擎的福,在作为后端服务时有比较高的运行效率,在很多场景下对于我们的日常开发足够用了。不过,它还为开发者开了一个使用C++ 开发 Node.js 原生扩展的口子,让开发者进行项目开发时有了更多的选择。 《Node.js:来一打 C++ 扩展》以 Chrome V8 的知识作为基础,配合 GYP 的一些内容,将教会大家如何使用 Node......一起来看看 《Node.js:来一打 C++ 扩展》 这本书的介绍吧!

HTML 压缩/解压工具
HTML 压缩/解压工具

在线压缩/解压 HTML 代码

SHA 加密
SHA 加密

SHA 加密工具

HSV CMYK 转换工具
HSV CMYK 转换工具

HSV CMYK互换工具