golang+websocket实现

话不多说直接上代码。

golang代码:

package main

import (
	"fmt"
	"net/http"
	"os"
	"reflect"
	"time"
	"unsafe"

	"github.com/gorilla/websocket"
)


type HttpHandler struct {
	http.Handler
}

func main() {
	var httpHandler HttpHandler
	http.Handle("/connect", httpHandler)
	if err := http.ListenAndServe(":9999", nil); err != nil {
		fmt.Println("程序退出")
		os.Exit(1)
	}
}

func (httpHandler HttpHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
	fmt.Println("程序进入监听>>")
	var upgrader = websocket.Upgrader{
		//解决跨域问题
		CheckOrigin: func(r *http.Request) bool {
			return true
		},
	}
	conn, err := upgrader.Upgrade(w, r, nil)
	if err != nil {
		fmt.Println("upgrade error:", err)
		return
	}
	defer conn.Close()
	conn.SetReadDeadline(time.Now().Add(6*time.Second))
	conn.SetPongHandler(func(string) error {
		fmt.Println("接收心跳响应<<")
		conn.SetReadDeadline(time.Now().Add(6*time.Second))
		return nil
	})
	ticker := time.NewTicker(2*time.Second)
	go func() {
		for {
			//从定时器中获取数据
			_ = <-ticker.C
			fmt.Println("发送心跳包>>")
			conn.WriteMessage(websocket.PingMessage, []byte{})
		}
	}()
	defer ticker.Stop()
	for {
		messageType, message, err := conn.ReadMessage()
		if err != nil {
			fmt.Println("接收异常:", err)
			break
		}
		fmt.Println(fmt.Sprintf("接收消息内容 >>%s", message))
		respMessage := fmt.Sprintf("I am Server, %s.", time.Now().Format("2006-01-02 15:04:05"))
		err = conn.WriteMessage(messageType, String2Bytes(respMessage))
		if err != nil {
			fmt.Println("发送异常:", err)
			break
		}
	}
}

func String2Bytes(s string) []byte {
	sh := (*reflect.StringHeader)(unsafe.Pointer(&s))
	bh := reflect.SliceHeader{
		Data: sh.Data,
		Len:  sh.Len,
		Cap:  sh.Len,
	}
	return *(*[]byte)(unsafe.Pointer(&bh))
}

web代码:

<html>
	<head>
		<meta charset="UTF-8"/>
	head>
	<body>
		<div>
			<button onclick="start()" style="width:80px;height:30px;line-height:24px;">开始button>
			<button onclick="stop()" style="width:80px;height:30px;line-height:24px;">停止button>
			<button onclick="reset()" style="width:80px;height:30px;line-height:24px;">重置button>
		div>
		<div id="connect-show">未连接div>
		<div id="rec-show">div>
	body>
	<script type="text/javascript">
		var ws;
		var interval;
	
		function rewebsocket() {
			//ws = new WebSocket("ws://127.0.0.1:9999/connect");
			ws = new WebSocket("ws://192.168.3.47:9999/connect");
			//连接建立时触发
			ws.onopen = function() {
				document.getElementById("connect-show").innerHTML = "连接成功";
				send(ws);
			}
			//连接关闭时触发
			ws.onclose = function() {
				document.getElementById("connect-show").innerHTML = "连接已关闭";
			}
			//通信发生错误时触发
			ws.onerror = function() {
				document.getElementById("connect-show").innerHTML = "连接通讯成功";
			}
			//客户端接收服务端数据时触发
			ws.onmessage = function(e) {
				var recMsg = e.data;
				var element = document.getElementById("rec-show");
				var nodes = element.childNodes;
				if (nodes && nodes.length >= 20) {
					element.removeChild(nodes[nodes.length-1])
				}
				element.innerHTML = ("
" + recMsg + "
"
+ element.innerHTML); } } //启动 function start() { if (ws == undefined || ws.readyState >= 2) { rewebsocket(); } } //停止 function stop() { clearInterval(interval); closeWebsocket(ws); interval = undefined; ws = undefined; } //重置 function reset() { stop(); document.getElementById("connect-show").innerHTML = "未连接"; document.getElementById("rec-show").innerHTML = ""; start(); } //消息发送策略 function send(ws) { interval = setInterval(() => { ws.send("I am client, " + getTime() + ".") },1000); } //获取客户端年月日时分秒 function getTime() { return new Date( +new Date() + 8 * 3600 * 1000 ).toJSON().substr(0,19).replace("T"," "); } //关闭websocket连接 function closeWebsocket(ws) { try { if (ws != undefined && ws.readyState < 2) { ws.close(); } } catch (err){} }
script> html>

以上示例中使用服务端心跳方式实现健康检查,也可以改为客户端心跳方式。

你可能感兴趣的:(Golang,golang,websocket)