基于 golang websocket 的聊天室 demo

本贴最后更新于 1954 天前,其中的信息可能已经时异事殊

测试聊天室 demo

具体代码来自github.com

main.go(入口函数)

package main  
  
import (  
  "log"  
 "net/http")  
  
func serveHome(w http.ResponseWriter, r *http.Request) {  
  if r.Method != "GET" {  
  http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)  
  return  
  }  
  http.ServeFile(w, r, "home.html")  
}  
  
func main() {  
 hub := newHub()  
  //开启一个调度器  
  go hub.run()  
  //demo页面  
  http.HandleFunc("/", serveHome)  
  //websocket 握手  
  http.HandleFunc("/ws", func(w http.ResponseWriter, r *http.Request) {  
  serveWs(hub, w, r)  
 }) err := http.ListenAndServe(":9017", nil)  
  if err != nil {  
  log.Fatal("ListenAndServe: ", err)  
 }}

hub.go(调度代码,通过 map 对 client 的维护,以及消息的广播 )

package main  
  
import "log"  
  
type Hub struct {  
  //用于维护用户的map  
  clients map[*Client]bool  
  
  //用于广播给用户的chan  
  broadcast chan []byte  
  
  //用于用户订阅的chan  
  register chan *Client  
  
  //用于用户取消订阅的chan  
  unregister chan *Client  
}  
  
//实例化一个调度器  
func newHub() *Hub {  
  return &Hub{  
 broadcast:  make(chan []byte),  
  register:   make(chan *Client),  
  unregister: make(chan *Client),  
  clients:    make(map[*Client]bool),  
  }  
}  
  
//开始运行调度器  
func (h *Hub) run() {  
  
  for {  
  select {  
  case client := <-h.register:  
         log.Printf("客户端 %d: 订阅\n", client.id)  
 h.clients[client] = true  
  case client := <-h.unregister:  
         log.Printf("客户端 %d: 取消订阅\n", client.id)  
  if _, ok := h.clients[client]; ok {  
  delete(h.clients, client)  
  close(client.send)  
 }  case message := <-h.broadcast:  
         log.Println("通过调度器把数据塞给client的chan,然后client.writePump广播消息")  
  for client := range h.clients {  
  select {  
  case client.send <- message:  
               log.Printf("将消息发送给客户端%d\n", client.id)  
  default:  
               close(client.send)  
  delete(h.clients, client)  
 } } } }}

client.go(client 信息的发送与接收)

package main  
  
import (  
  "bytes"  
 "log" "net/http" "time"  
 "github.com/gorilla/websocket")  
  
var iii int  
  
const (  
  //写入数据允许的等待时间  
  writeWait = 10 * time.Second  
  
  //pong的周期  
  pongWait = 60 * time.Second  
  
  //ping的周期 (需要小于pong的周期)  
  pingPeriod = (pongWait * 9) / 10  
  
  //消息最大的字节数  
  maxMessageSize = 512  
)  
  
var (  
 newline = []byte{'\n'}  
 space = []byte{' '}  
)  
  
var upgrader = websocket.Upgrader{  
 ReadBufferSize:  1024,  
  WriteBufferSize: 1024,  
  //允许跨域  
  CheckOrigin: func(r *http.Request) bool {  
  return true  
  },  
}  
  
// Client is a middleman between the websocket connection and the hub.  
type Client struct {  
  //客户端的调度器  
  hub *Hub  
  
  //每个client实例化一个链接  
  conn *websocket.Conn  
  
  //从调度器中的向客户端发送数据  
  send chan []byte  
  
  //每个客户端的唯一标识  
  id int  
}  
  
//从链接中进行读取  
func (c *Client) readPump() {  
  defer func() {  
 c.hub.unregister <- c  
 c.conn.Close()  
 }()  log.Println("设置读取信息的最大值")  
 c.conn.SetReadLimit(maxMessageSize)  
 c.conn.SetReadDeadline(time.Now().Add(pongWait))  
 c.conn.SetPongHandler(func(string) error { c.conn.SetReadDeadline(time.Now().Add(pongWait)); return nil })  
  
  for {  
  log.Println("读取连接中的信息")  
 _, message, err := c.conn.ReadMessage()  
  if err != nil {  
  if websocket.IsUnexpectedCloseError(err, websocket.CloseGoingAway, websocket.CloseAbnormalClosure) {  
  log.Printf("error: %v", err)  
 }  return  
  }  
  log.Println("去除消息中的空格和换行")  
 message = bytes.TrimSpace(bytes.Replace(message, newline, space, -1))  
 c.hub.broadcast <- message  
 }}  
  
//从chan中获取信息进行写入  
func (c *Client) writePump() {  
 ticker := time.NewTicker(pingPeriod)  
  defer func() {  
 ticker.Stop()  
 c.conn.Close()  
 }()  for {  
  select {  
  
  case message, ok := <-c.send:  
         log.Println("writePump 接收到调度器send message")  
 c.conn.SetWriteDeadline(time.Now().Add(writeWait))  
  if !ok {  
 c.conn.WriteMessage(websocket.CloseMessage, []byte{})  
  return  
  }  
  log.Println("指定下一条发送消息的类型")  
 w, err := c.conn.NextWriter(websocket.TextMessage)  
  if err != nil {  
  return  
  }  
 w.Write(message)  
  
  // Add queued chat messages to the current websocket message.  
  n := len(c.send)  
  for i := 0; i < n; i++ {  
 w.Write(newline)  
 w.Write(<-c.send)  
 }  
  if err := w.Close(); err != nil {  
  return  
  }  
  case <-ticker.C:  
         c.conn.SetWriteDeadline(time.Now().Add(writeWait))  
  log.Println("接收到定时器的心跳ping-pong")  
  if err := c.conn.WriteMessage(websocket.PingMessage, nil); err != nil {  
  return  
  }  
 } }}  
  
//每新增一个连接初始化一个客户端进行维护  
func serveWs(hub *Hub, w http.ResponseWriter, r *http.Request) {  
  
 conn, err := upgrader.Upgrade(w, r, nil)  
  if err != nil {  
  log.Println(err)  
  return  
  }  
  //用于查看用户(测试的假数据)  
  iii++  
  log.Println("初始化一个新的客户端")  
 client := &Client{hub: hub, conn: conn, send: make(chan []byte, 256), id: iii}  
 client.hub.register <- client  
  
  //客户端写入  
  log.Println("开启client.writePump")  
  go client.writePump()  
  
  log.Println("开启client.readPump")  
  //客户端读取  
  go client.readPump()  
}
<html>
<head>
    <title>Chat Example<title>
    <script>
   window.onload = function () {  
  var conn;  
 var msg = document.getElementById("msg");  
 var log = document.getElementById("log");  
  
 function appendLog(item) {  
  var doScroll = log.scrollTop \> log.scrollHeight \- log.clientHeight \- 1;  
  log.appendChild(item);  
 if (doScroll) {  
  log.scrollTop = log.scrollHeight \- log.clientHeight;  
  }  
 }  
  document.getElementById("form").onsubmit = function () {  
  if (!conn) {  
  return false;  
  }  
  if (!msg.value) {  
  return false;  
  }  
  conn.send(msg.value);  
  msg.value = "";  
 return false;  };  
 if (window\["WebSocket"\]) {  
  conn = new WebSocket("ws://123.207.1.120:9017/ws");  
  conn.onclose = function (evt) {  
  var item = document.createElement("div");  
  item.innerHTML = "Connection closed.";  
  appendLog(item);  
  };  
  conn.onmessage = function (evt) {  
  var messages = evt.data.split('\\n');  
 for (var i = 0; i < messages.length; i++) {  
  var item = document.createElement("div");  
  item.innerText = messages\[i\];  
  appendLog(item);  
  }  
 };  
  } else {  
  var item = document.createElement("div");  
  item.innerHTML = "Your browser does not support WebSockets.";  
  appendLog(item);  
  }  
};
    </script>
<style type="text/css">
html {  
  overflow: hidden;  
}  
  
body {  
  overflow: hidden;  
  padding: 0;  
  margin: 0;  
  width: 100%;  
  height: 100%;  
  background: gray;  
}  
  
#log {  
  background: white;  
  margin: 0;  
  padding: 0.5em 0.5em 0.5em 0.5em;  
  position: absolute;  
  top: 0.5em;  
  left: 0.5em;  
  right: 0.5em;  
  bottom: 3em;  
  overflow: auto;  
}  
  
#form {  
  padding: 0 0.5em 0 0.5em;  
  margin: 0;  
  position: absolute;  
  bottom: 1em;  
  left: 0px;  
  width: 100%;  
  overflow: hidden;  
}
</style>
<body>
  <div id="log"></div>
  <form id="form">
  <input type="submit" value="Send"/>
  <input type="text" id="msg" size="64">
  </form>
</body>

</html>



  • golang

    Go 语言是 Google 推出的一种全新的编程语言,可以在不损失应用程序性能的情况下降低代码的复杂性。谷歌首席软件工程师罗布派克(Rob Pike)说:我们之所以开发 Go,是因为过去 10 多年间软件开发的难度令人沮丧。Go 是谷歌 2009 发布的第二款编程语言。

    492 引用 • 1384 回帖 • 363 关注
  • WebSocket

    WebSocket 是 HTML5 中定义的一种新协议,它实现了浏览器与服务器之间的全双工通信(full-duplex)。

    48 引用 • 206 回帖 • 380 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
  • xhaoxiong

    刚初步入手关于 golang 网络编程方面的一些知识,感谢指点

  • 其他回帖
  • broadcast 和 client.send 这两个 chan 建议加上缓冲,不然会造成大量阻塞甚至死锁

    这个 websocket 包虽然有自动 pingpong 功能,但是建议加上客户端 ping 服务端然后重连的功能

    1 回复
xhaoxiong
站在巨人的肩膀上学习与创新

推荐标签 标签

  • Dubbo

    Dubbo 是一个分布式服务框架,致力于提供高性能和透明化的 RPC 远程服务调用方案,是 [阿里巴巴] SOA 服务化治理方案的核心框架,每天为 2,000+ 个服务提供 3,000,000,000+ 次访问量支持,并被广泛应用于阿里巴巴集团的各成员站点。

    60 引用 • 82 回帖 • 609 关注
  • SMTP

    SMTP(Simple Mail Transfer Protocol)即简单邮件传输协议,它是一组用于由源地址到目的地址传送邮件的规则,由它来控制信件的中转方式。SMTP 协议属于 TCP/IP 协议簇,它帮助每台计算机在发送或中转信件时找到下一个目的地。

    4 引用 • 18 回帖 • 594 关注
  • uTools

    uTools 是一个极简、插件化、跨平台的现代桌面软件。通过自由选配丰富的插件,打造你得心应手的工具集合。

    5 引用 • 13 回帖
  • SpaceVim

    SpaceVim 是一个社区驱动的模块化 vim/neovim 配置集合,以模块的方式组织管理插件以
    及相关配置,为不同的语言开发量身定制了相关的开发模块,该模块提供代码自动补全,
    语法检查、格式化、调试、REPL 等特性。用户仅需载入相关语言的模块即可得到一个开箱
    即用的 Vim-IDE。

    3 引用 • 31 回帖 • 81 关注
  • Laravel

    Laravel 是一套简洁、优雅的 PHP Web 开发框架。它采用 MVC 设计,是一款崇尚开发效率的全栈框架。

    19 引用 • 23 回帖 • 693 关注
  • Kotlin

    Kotlin 是一种在 Java 虚拟机上运行的静态类型编程语言,由 JetBrains 设计开发并开源。Kotlin 可以编译成 Java 字节码,也可以编译成 JavaScript,方便在没有 JVM 的设备上运行。在 Google I/O 2017 中,Google 宣布 Kotlin 成为 Android 官方开发语言。

    19 引用 • 33 回帖 • 29 关注
  • frp

    frp 是一个可用于内网穿透的高性能的反向代理应用,支持 TCP、UDP、 HTTP 和 HTTPS 协议。

    15 引用 • 7 回帖 • 3 关注
  • 学习

    “梦想从学习开始,事业从实践起步” —— 习近平

    162 引用 • 473 回帖
  • OpenStack

    OpenStack 是一个云操作系统,通过数据中心可控制大型的计算、存储、网络等资源池。所有的管理通过前端界面管理员就可以完成,同样也可以通过 Web 接口让最终用户部署资源。

    10 引用 • 3 关注
  • 支付宝

    支付宝是全球领先的独立第三方支付平台,致力于为广大用户提供安全快速的电子支付/网上支付/安全支付/手机支付体验,及转账收款/水电煤缴费/信用卡还款/AA 收款等生活服务应用。

    29 引用 • 347 回帖 • 3 关注
  • TextBundle

    TextBundle 文件格式旨在应用程序之间交换 Markdown 或 Fountain 之类的纯文本文件时,提供更无缝的用户体验。

    1 引用 • 2 回帖 • 49 关注
  • ZooKeeper

    ZooKeeper 是一个分布式的,开放源码的分布式应用程序协调服务,是 Google 的 Chubby 一个开源的实现,是 Hadoop 和 HBase 的重要组件。它是一个为分布式应用提供一致性服务的软件,提供的功能包括:配置维护、域名服务、分布式同步、组服务等。

    59 引用 • 29 回帖 • 23 关注
  • 以太坊

    以太坊(Ethereum)并不是一个机构,而是一款能够在区块链上实现智能合约、开源的底层系统。以太坊是一个平台和一种编程语言 Solidity,使开发人员能够建立和发布下一代去中心化应用。 以太坊可以用来编程、分散、担保和交易任何事物:投票、域名、金融交易所、众筹、公司管理、合同和知识产权等等。

    34 引用 • 367 回帖 • 1 关注
  • CodeMirror
    1 引用 • 2 回帖 • 119 关注
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    90 引用 • 383 回帖
  • 程序员

    程序员是从事程序开发、程序维护的专业人员。

    539 引用 • 3528 回帖
  • React

    React 是 Facebook 开源的一个用于构建 UI 的 JavaScript 库。

    192 引用 • 291 回帖 • 440 关注
  • OpenShift

    红帽提供的 PaaS 云,支持多种编程语言,为开发人员提供了更为灵活的框架、存储选择。

    14 引用 • 20 回帖 • 607 关注
  • PostgreSQL

    PostgreSQL 是一款功能强大的企业级数据库系统,在 BSD 开源许可证下发布。

    22 引用 • 22 回帖
  • Windows

    Microsoft Windows 是美国微软公司研发的一套操作系统,它问世于 1985 年,起初仅仅是 Microsoft-DOS 模拟环境,后续的系统版本由于微软不断的更新升级,不但易用,也慢慢的成为家家户户人们最喜爱的操作系统。

    215 引用 • 462 回帖
  • jQuery

    jQuery 是一套跨浏览器的 JavaScript 库,强化 HTML 与 JavaScript 之间的操作。由 John Resig 在 2006 年 1 月的 BarCamp NYC 上释出第一个版本。全球约有 28% 的网站使用 jQuery,是非常受欢迎的 JavaScript 库。

    63 引用 • 134 回帖 • 740 关注
  • WiFiDog

    WiFiDog 是一套开源的无线热点认证管理工具,主要功能包括:位置相关的内容递送;用户认证和授权;集中式网络监控。

    1 引用 • 7 回帖 • 546 关注
  • Chrome

    Chrome 又称 Google 浏览器,是一个由谷歌公司开发的网页浏览器。该浏览器是基于其他开源软件所编写,包括 WebKit,目标是提升稳定性、速度和安全性,并创造出简单且有效率的使用者界面。

    60 引用 • 287 回帖
  • Flutter

    Flutter 是谷歌的移动 UI 框架,可以快速在 iOS 和 Android 上构建高质量的原生用户界面。 Flutter 可以与现有的代码一起工作,它正在被越来越多的开发者和组织使用,并且 Flutter 是完全免费、开源的。

    39 引用 • 92 回帖 • 4 关注
  • IDEA

    IDEA 全称 IntelliJ IDEA,是一款 Java 语言开发的集成环境,在业界被公认为最好的 Java 开发工具之一。IDEA 是 JetBrains 公司的产品,这家公司总部位于捷克共和国的首都布拉格,开发人员以严谨著称的东欧程序员为主。

    180 引用 • 400 回帖 • 1 关注
  • CloudFoundry

    Cloud Foundry 是 VMware 推出的业界第一个开源 PaaS 云平台,它支持多种框架、语言、运行时环境、云平台及应用服务,使开发人员能够在几秒钟内进行应用程序的部署和扩展,无需担心任何基础架构的问题。

    5 引用 • 18 回帖 • 156 关注
  • Kafka

    Kafka 是一种高吞吐量的分布式发布订阅消息系统,它可以处理消费者规模的网站中的所有动作流数据。 这种动作(网页浏览,搜索和其他用户的行动)是现代系统中许多功能的基础。 这些数据通常是由于吞吐量的要求而通过处理日志和日志聚合来解决。

    35 引用 • 35 回帖