一百行Golang代碼實現(xiàn)簡單并發(fā)聊天室
項目介紹:Golang100行代碼實現(xiàn)高并發(fā)聊天室,其中實現(xiàn)的功能有:上下線廣播,私聊,用戶改名,超時強(qiáng)踢,在線用戶檢測等
在開始項目前,我們需要理解貫穿這整個項目的兩個重要變量,若能理解這兩個變量的使用,那么并發(fā)聊天室項目會變得手到擒來。第一個是onlinemap全局map,第二個是Message全局channel。
取名為onlinemap的全局map類型為map[string][client],這個全局字典是用來存儲當(dāng)前在此聊天室的用戶的,key值是string類型,為用戶的ip地址+Port端口,對應(yīng)的value值為一個結(jié)構(gòu)體,結(jié)構(gòu)體內(nèi)有此用戶的姓名,地址和管道(用來給每一個用戶傳輸信息,服務(wù)于Message全局通道)
取名為Message的全局channel也貫穿在整段代碼中,向其中傳送數(shù)據(jù)時,Message會在另一個go程里向其他每一個在線用戶的管道中發(fā)送內(nèi)容,隨后在另一個go程里每一個用戶的管道會向?qū)?yīng)用戶轉(zhuǎn)發(fā)內(nèi)容。如此可以實現(xiàn)上下線廣播,群聊的功能。而每一個用戶私有的管道可以實現(xiàn)私聊功能。
這個圖詳細(xì)闡述了這段代碼的工作流程。

理解以上內(nèi)容 下面我們再來看代碼,就會很輕松,如果還是一頭霧水也不要著急,小編會在下面每一行代碼都加上精準(zhǔn)通俗的注釋,不多說,上代碼。
package main
import (
"net"
"fmt"
"strings"
"time"
)
//定義的此結(jié)構(gòu)體為全局map的value值,包括每一個用戶的姓名,ip地址和私人管道
type client struct {
name string
addr string
C chan string
}
/*這個函數(shù)是將私人管道中的內(nèi)容發(fā)送給用戶,配合全局管道Message使用可以實現(xiàn)廣播的功能,
單獨使用可以實現(xiàn)私聊的功能*/
func writemsg2client(clinet client,conn net.Conn) {
for m := range clinet.C {
conn.Write([]byte(m + "\n"))
}
}
//這只是一個封裝好用來統(tǒng)一(發(fā)送信息格式)的小函數(shù),不用在意
func makemsg(name string, addr string, s string) string {
return "[" + addr + "]" + name + s
}
//每一個進(jìn)入聊天室的用戶都將啟動一個handleconn的go程來處理事件
func handleconn(conn net.Conn) {
defer conn.Close()
/*用戶連接進(jìn)來以后要初始化全局map,把自己的信息加入到字典里,相當(dāng)于進(jìn)到聊天室里之前要登
記一下個人信息,注意姓名初始為ip地址。*/
addr := conn.RemoteAddr().String()
fmt.Printf("用戶%s進(jìn)入了房間\n", addr)
client := client{addr, addr, make(chan string)}
//在這里啟動子go程,功能上面已經(jīng)提及
go writemsg2client(client,conn)
onlinemap[addr] = client
//登錄進(jìn)來一切準(zhǔn)備就緒后就給所有人廣播上線信息啦
Message <- makemsg(client.name, addr, "login")
//下面這三個變量服務(wù)于下面一些小功能
var haschat=make(chan bool)
var ifquit=make(chan bool)
var flag bool
//從這單獨開啟一個go程來讀取用戶輸入的信息
go func() {
buf:=make([]byte,4096)
for {
n,_:=conn.Read(buf)
if n==0 {
fmt.Printf("%s離開了房間\n",client.name)
ifquit<-true
return
}
//改名功能的實現(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"{
//查詢在線用戶信息的功能
for _,s:=range onlinemap{
conn.Write([]byte(s.name+"online\n"))
}
}else if string(buf[:2])=="m|"&&strings.Count(string(buf[:n]),"|")==2 {
/*私聊功能的實現(xiàn),其實私聊功能就是跳過了往全局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:
//超時強(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
}
}
}
//這個函數(shù)用來將全局Message中的內(nèi)容全部塞到私人管道C里,實現(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
}
//每一個連接進(jìn)來的用戶都會被分配進(jìn)入一個子go程,用來處理上面我們提到的各種功能
go handleconn(conn)
}
}
以上就是一個簡單的高并發(fā)聊天室了,依托于go語言的強(qiáng)大,去掉注釋只剩下不到一百行,雖然功能簡單,但是涉及到channel,socket,select,map,string及go的使用,有利于此階段在學(xué)的小伙伴們學(xué)習(xí)交流,大家有什么疑問或者想法可以在下面給我留言哦。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Golang中unicode碼和中文的互相轉(zhuǎn)換函數(shù)使用
這篇文章主要為大家介紹了Golang中unicode碼和中文的互相轉(zhuǎn)換函數(shù)使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09
Go實現(xiàn)生產(chǎn)隨機(jī)密碼的示例代碼
這篇文章主要為大家詳細(xì)介紹了如何利用Go實現(xiàn)生產(chǎn)隨機(jī)密碼的,文中的示例代碼簡潔易懂,具有一定的借鑒價值,有需要的小伙伴可以參考一下2023-09-09
Go 并發(fā)實現(xiàn)協(xié)程同步的多種解決方法
這篇文章主要介紹了Go 并發(fā)——實現(xiàn)協(xié)程同步的多種解決方法,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下2018-08-08

