Go語言實現(xiàn)一個簡單的并發(fā)聊天室的項目實戰(zhàn)
寫在前面
Go語言在很多方面天然的具備很多便捷性,譬如網(wǎng)絡(luò)編程,并發(fā)編程。而通道則又是Go語言實現(xiàn)并發(fā)編程的重要工具,因為其承擔著通道之間互相通信的重任。并且因為其本身就是并發(fā)安全的,所以在某些場景下是非常好用的。
并發(fā)聊天服務(wù)器
這里主要是實現(xiàn)一個簡單的并發(fā)聊天服務(wù)器。首先,客戶端可以在服務(wù)器中注冊自己的信息(登錄以及退出),客戶端發(fā)出的所有的信息由服務(wù)器向各個客戶端進行轉(zhuǎn)發(fā),或者換句話說是廣播。
具體代碼
服務(wù)端
說的再多,沒有代碼簡單明了,直接上代碼~
package main
import (
?? ?"bufio"
?? ?"fmt"
?? ?"log"
?? ?"net"
)
type client chan <- string //定義一個單向的向外發(fā)送數(shù)據(jù)的通道
var (
?? ?entering = make(chan client)
?? ?leaving = make(chan client)
?? ?messages = make(chan string)
)
func main() {
?? ?listener, err := net.Listen("tcp","localhost:8000")
?? ?if err != nil {
?? ??? ?log.Fatal("network is broken", err)
?? ?}
?? ?go broadcaster()
?? ?for {
?? ??? ?conn, err := listener.Accept()
?? ??? ?if err != nil {
?? ??? ??? ?log.Print(err)
?? ??? ??? ?continue
?? ??? ?}
?? ??? ?go handleConn1(conn)
?? ?}
}
func broadcaster() ?{
?? ?clients := make(map[client]bool) //存儲每個client的登錄狀態(tài)
?? ?for{
?? ??? ?select {
?? ??? ?case msg := <-messages:
?? ??? ??? ?for cli := range clients {
?? ??? ??? ??? ?cli <- msg
?? ??? ??? ?}
?? ??? ?case cli := <-entering:
?? ??? ??? ?clients[cli] = true
?? ??? ?case cli := <-leaving:
?? ??? ??? ?delete(clients,cli)
?? ??? ??? ?close(cli)
?? ??? ?}
?? ?}
}
func handleConn1(conn net.Conn) ?{
?? ?ch := make(chan string)
?? ?go clientWriter(conn, ch)
?? ?who := conn.RemoteAddr().String()
?? ?ch <- "You are " + who
?? ?entering <- ch
?? ?messages <- who + "has arrived"
?? ?input := bufio.NewScanner(conn)
?? ?for input.Scan() {
?? ??? ?messages <- who + ":" + input.Text()
?? ?}
?? ?leaving <- ch
?? ?messages <- who + "has left"
?? ?conn.Close()
}
func clientWriter(conn net.Conn, ch <- chan string) ?{
?? ?for msg := range ch {
?? ??? ?fmt.Fprintln(conn, msg)
?? ?}
}客戶端
客戶端相對簡單,只是涉及到信息的發(fā)送和接受工作。
package main
import (
?? ?"io"
?? ?"log"
?? ?"net"
?? ?"os"
)
func main() {
?? ?conn, err := net.Dial("tcp","localhost:8000")
?? ?if err != nil {
?? ??? ?log.Fatal("Connected has been refused!",err)
?? ?}
?? ?defer conn.Close()
?? ?go mesCopy(os.Stdout,conn)
?? ?mesCopy(conn,os.Stdin)
}
func mesCopy(des io.Writer, res io.Reader) ?{
?? ?if _, err := io.Copy(des, res); err != nil {
?? ??? ?log.Print("wrong!")
?? ?}
}總結(jié)
實現(xiàn)原理較為簡單,所以代碼并沒有多少注釋,如果有任何疑問,歡迎留言討論。最后說一句,在MIT的課程中,其實并不是很推薦在并發(fā)編程中使用通道chan,除非你對其應(yīng)用的場景和可能出現(xiàn)的情況有很好的把握,不然可能會出現(xiàn)很多不可預(yù)測的事情,譬如死鎖(見另外一篇博客)。在這種時候,共享變量將會是一種很好的選擇,具體查看go多線程實踐。
到此這篇關(guān)于Go語言實現(xiàn)一個簡單的并發(fā)聊天室的項目實戰(zhàn)的文章就介紹到這了,更多相關(guān)Go語言 并發(fā)聊天室內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go中的new()和make()函數(shù)區(qū)別及底層原理詳解
這篇文章主要為大家介紹了Go中的new()和make()函數(shù)區(qū)別及底層原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-09-09
Golang Socket Server自定義協(xié)議的簡單實現(xiàn)方案
這篇文章主要介紹了Golang Socket Server自定義協(xié)議的簡單實現(xiàn)方案,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12
Golang 如何判斷數(shù)組某個元素是否存在 (isset)
這篇文章主要介紹了Golang 如何判斷數(shù)組某個元素是否存在 (isset),具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-04-04

