網站首頁 編程語言 正文
使用socket和channel,實現(xiàn)簡單控制臺聊天室
這里使用socket和channel,演示在GO中如何編寫一個簡單網絡程序
功能分析
聊天室主要功能:用戶可以加入/離開聊天室;每個用戶發(fā)送的消息,廣播給所有人
聊天室分為客戶端和服務端,客戶端負責發(fā)送消息和打印服務器消息,服務器負責接收客戶端消息,并廣播給所有人
客戶端可以使用telnet程序
服務端是需要實現(xiàn)的。需要實現(xiàn)的功能,
- 如何保存多個客戶端的連接,管理連接的接入與斷開
- 如何接收和廣播客戶端消息
實現(xiàn)思路
通過功能分析,拆分為聊天室結構體和客戶端結構體
聊天室結構體負責管理當前接入的客戶端和廣播消息
客戶端結構體負責管理socket連接和需要接收與發(fā)送的數(shù)據
客戶端連接/斷開時通知聊天室;客戶端發(fā)送的消息實際是轉發(fā)給聊天室,然后聊天室再廣播出去
完整代碼
package main import ( "bufio" "fmt" "log" "net" ) type Client struct { id string conn *net.Conn message chan string } type Hub struct { clients map[*Client]bool entering chan *Client leaving chan *Client messages chan string } func main() { hub := &Hub{ clients: make(map[*Client]bool), entering: make(chan *Client), leaving: make(chan *Client), messages: make(chan string), } listener, err := net.Listen("tcp", ":8000") if err != nil { log.Fatal(err) } go hub.broadcaster() for { conn, err := listener.Accept() if err != nil { log.Println(err) continue } go hub.handleConn(conn) } } func (hub *Hub) broadcaster() { for { select { case msg := <-hub.messages: for cli := range hub.clients { cli.message <- msg } case cli := <-hub.entering: hub.clients[cli] = true case cli := <-hub.leaving: delete(hub.clients, cli) } } } func (hub *Hub) handleConn(conn net.Conn) { defer conn.Close() ch := make(chan string) who := conn.RemoteAddr().String() client := &Client{who, &conn, ch} go hub.writeLoop(client) ch <- "welcome " + client.id hub.messages <- client.id + " join chat" hub.entering <- client hub.readLoop(client) hub.messages <- client.id + " has left" hub.leaving <- client } func (hub *Hub) writeLoop(client *Client) { for msg := range client.message { fmt.Fprintf(*client.conn, "%s\n", msg) } } func (hub *Hub) readLoop(client *Client) { input := bufio.NewScanner(*client.conn) for input.Scan() { hub.messages <- client.id + ": " + input.Text() } }
分析
實現(xiàn)的關鍵是封裝了客戶端通信channel,無論是遠程發(fā)送過來的消息還是聊天室廣播的消息,都通過這個channel傳遞,且這個channel是綁定客戶端的
參考鏈接中,直接使用channel來定義客戶端type client chan<- string
,其實更能表達這一點
為了容易理解,這里將channel封裝為客戶端的一個通信管道,客戶端還可以有別的屬性,例如:id、連接和超時時間等
參考: Go 網絡編程示例
原文鏈接:https://www.cnblogs.com/snowsteps/p/15711223.html
相關推薦
- 2022-08-12 Python中深拷貝與淺拷貝的區(qū)別介紹_python
- 2022-02-05 lxml:提取html標簽中的內容
- 2022-06-11 python?針對在子文件夾中的md文檔實現(xiàn)批量md轉word_python
- 2022-06-26 ASP.NET?Core中引用OpenAPI服務的添加示例_實用技巧
- 2023-10-30 springboot application 常用配置
- 2023-04-04 Swift中的HTTP請求體Request?Bodies使用示例詳解_IOS
- 2022-12-08 Python中的pandas庫簡介及其使用教程_python
- 2022-06-02 C語言如何讀取bmp圖像_C 語言
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學習環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據結構-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支