一百行Golang代碼實(shí)現(xiàn)簡(jiǎn)單并發(fā)聊天室
項(xiàng)目介紹:Golang100行代碼實(shí)現(xiàn)高并發(fā)聊天室,其中實(shí)現(xiàn)的功能有:上下線廣播,私聊,用戶改名,超時(shí)強(qiáng)踢,在線用戶檢測(cè)等
在開始項(xiàng)目前,我們需要理解貫穿這整個(gè)項(xiàng)目的兩個(gè)重要變量,若能理解這兩個(gè)變量的使用,那么并發(fā)聊天室項(xiàng)目會(huì)變得手到擒來。第一個(gè)是onlinemap全局map,第二個(gè)是Message全局channel。
取名為onlinemap的全局map類型為map[string][client],這個(gè)全局字典是用來存儲(chǔ)當(dāng)前在此聊天室的用戶的,key值是string類型,為用戶的ip地址+Port端口,對(duì)應(yīng)的value值為一個(gè)結(jié)構(gòu)體,結(jié)構(gòu)體內(nèi)有此用戶的姓名,地址和管道(用來給每一個(gè)用戶傳輸信息,服務(wù)于Message全局通道)
取名為Message的全局channel也貫穿在整段代碼中,向其中傳送數(shù)據(jù)時(shí),Message會(huì)在另一個(gè)go程里向其他每一個(gè)在線用戶的管道中發(fā)送內(nèi)容,隨后在另一個(gè)go程里每一個(gè)用戶的管道會(huì)向?qū)?yīng)用戶轉(zhuǎn)發(fā)內(nèi)容。如此可以實(shí)現(xiàn)上下線廣播,群聊的功能。而每一個(gè)用戶私有的管道可以實(shí)現(xiàn)私聊功能。
這個(gè)圖詳細(xì)闡述了這段代碼的工作流程。
理解以上內(nèi)容 下面我們?cè)賮砜创a,就會(huì)很輕松,如果還是一頭霧水也不要著急,小編會(huì)在下面每一行代碼都加上精準(zhǔn)通俗的注釋,不多說,上代碼。
package main import ( "net" "fmt" "strings" "time" ) //定義的此結(jié)構(gòu)體為全局map的value值,包括每一個(gè)用戶的姓名,ip地址和私人管道 type client struct { name string addr string C chan string } /*這個(gè)函數(shù)是將私人管道中的內(nèi)容發(fā)送給用戶,配合全局管道Message使用可以實(shí)現(xiàn)廣播的功能, 單獨(dú)使用可以實(shí)現(xiàn)私聊的功能*/ func writemsg2client(clinet client,conn net.Conn) { for m := range clinet.C { conn.Write([]byte(m + "\n")) } } //這只是一個(gè)封裝好用來統(tǒng)一(發(fā)送信息格式)的小函數(shù),不用在意 func makemsg(name string, addr string, s string) string { return "[" + addr + "]" + name + s } //每一個(gè)進(jìn)入聊天室的用戶都將啟動(dòng)一個(gè)handleconn的go程來處理事件 func handleconn(conn net.Conn) { defer conn.Close() /*用戶連接進(jìn)來以后要初始化全局map,把自己的信息加入到字典里,相當(dāng)于進(jìn)到聊天室里之前要登 記一下個(gè)人信息,注意姓名初始為ip地址。*/ addr := conn.RemoteAddr().String() fmt.Printf("用戶%s進(jìn)入了房間\n", addr) client := client{addr, addr, make(chan string)} //在這里啟動(dòng)子go程,功能上面已經(jīng)提及 go writemsg2client(client,conn) onlinemap[addr] = client //登錄進(jìn)來一切準(zhǔn)備就緒后就給所有人廣播上線信息啦 Message <- makemsg(client.name, addr, "login") //下面這三個(gè)變量服務(wù)于下面一些小功能 var haschat=make(chan bool) var ifquit=make(chan bool) var flag bool //從這單獨(dú)開啟一個(gè)go程來讀取用戶輸入的信息 go func() { buf:=make([]byte,4096) for { n,_:=conn.Read(buf) if n==0 { fmt.Printf("%s離開了房間\n",client.name) ifquit<-true return } //改名功能的實(shí)現(xiàn) if string(buf[:7])=="Rename|" { client.name=strings.Split(string(buf[:n-1]),"|")[1] onlinemap[addr]=client conn.Write([]byte("rename success\n")) }else if string(buf[:n-1])=="/who"{ //查詢?cè)诰€用戶信息的功能 for _,s:=range onlinemap{ conn.Write([]byte(s.name+"online\n")) } }else if string(buf[:2])=="m|"&&strings.Count(string(buf[:n]),"|")==2 { /*私聊功能的實(shí)現(xiàn),其實(shí)私聊功能就是跳過了往全局Message里傳輸信息, 改為直接向私人管道里傳輸信息*/ flag=false slice:=strings.Split(string(buf[:n-1]),"|") for _,a:=range onlinemap{ //遍歷所有在線用戶,向指定的用戶管道中發(fā)送信息 if a.name==slice[1]{ flag=true a.C<-makemsg(client.name,addr,slice[2]) conn.Write([]byte("send success")) } } if flag { conn.Write([]byte("no such man or not online")) } } else { Message<-makemsg(client.name,addr,string(buf[:n-1])) } haschat<-true } }() for { select { case <-haschat: //超時(shí)強(qiáng)踢 case <-time.After(time.Minute*3): delete(onlinemap,addr) Message<-makemsg(client.name,addr,"out time to leave") close(client.C) return case <-ifquit: //退出處理 delete(onlinemap,addr) Message<-makemsg(client.name,addr,"out time to leave") close(client.C) return } } } //這個(gè)函數(shù)用來將全局Message中的內(nèi)容全部塞到私人管道C里,實(shí)現(xiàn)上下線廣播和群聊的功能 func Manager() { for { msg := <-Message for _, s := range onlinemap { s.C <- msg } } } var Message = make(chan string) var onlinemap map[string]client = make(map[string]client) //主函數(shù) func main() { listener, _ := net.Listen("tcp", "127.0.0.1:6666") defer listener.Close() //提前開啟全局Message的go程,防止被阻塞 go Manager() for { conn, err := listener.Accept() if err != nil { fmt.Println("accept err", err) continue } //每一個(gè)連接進(jìn)來的用戶都會(huì)被分配進(jìn)入一個(gè)子go程,用來處理上面我們提到的各種功能 go handleconn(conn) } }
以上就是一個(gè)簡(jiǎn)單的高并發(fā)聊天室了,依托于go語言的強(qiáng)大,去掉注釋只剩下不到一百行,雖然功能簡(jiǎn)單,但是涉及到channel,socket,select,map,string及go的使用,有利于此階段在學(xué)的小伙伴們學(xué)習(xí)交流,大家有什么疑問或者想法可以在下面給我留言哦。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
go語言實(shí)現(xiàn)簡(jiǎn)易比特幣系統(tǒng)之交易簽名及校驗(yàn)功能
這篇文章主要介紹了go語言實(shí)現(xiàn)簡(jiǎn)易比特幣系統(tǒng)之交易簽名及校驗(yàn)功能,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-04-04一文詳解go中如何實(shí)現(xiàn)定時(shí)任務(wù)
定時(shí)任務(wù)是指按照預(yù)定的時(shí)間間隔或特定時(shí)間點(diǎn)自動(dòng)執(zhí)行的計(jì)劃任務(wù)或操作,這篇文章主要為大家詳細(xì)介紹了go中是如何實(shí)現(xiàn)定時(shí)任務(wù)的,感興趣的可以了解下2023-11-11解決Golang中ResponseWriter的一個(gè)坑
這篇文章主要介紹了解決Golang中ResponseWriter的一個(gè)坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2021-04-04Go?并發(fā)編程協(xié)程及調(diào)度機(jī)制詳情
這篇文章主要介紹了Go并發(fā)編程協(xié)程及調(diào)度機(jī)制詳情,協(xié)程是Go語言最大的特色之一,goroutine的實(shí)現(xiàn)其實(shí)是通過協(xié)程,更多相關(guān)內(nèi)容需要的朋友可以參考一下2022-09-09