Golang實(shí)現(xiàn)的聊天程序服務(wù)端和客戶端代碼分享
實(shí)現(xiàn)邏輯
1、Golang 版本 1.3
2、實(shí)現(xiàn)原理:
1、主進(jìn)程建立TCP監(jiān)聽(tīng)服務(wù),并且初始化一個(gè)變量 talkChan := make(map[int]chan string)
2、當(dāng)主進(jìn)程ACCEPT連接請(qǐng)求后,利用go 啟動(dòng)一個(gè)協(xié)程A去維持和客戶端的連接,把taokChan帶入到協(xié)程里
3、和客戶端建立連接的協(xié)程A,發(fā)送消息給客戶端,使其發(fā)送自己的用戶信息。
4、協(xié)程A在收到客戶端發(fā)送的用戶信息后,建立一個(gè)此用戶對(duì)應(yīng)的管道 talkChan[uid] = make(chan string)
5、協(xié)程A再啟動(dòng)一個(gè)協(xié)程A1去專(zhuān)門(mén)用來(lái)讀取客戶端發(fā)送的消息,并且用來(lái)判斷是發(fā)送給誰(shuí)的消息,然后把消息放到對(duì)應(yīng)的chan里。
6、協(xié)程A再啟動(dòng)一個(gè)協(xié)程A2用來(lái)讀取此用戶對(duì)應(yīng)的管道,如果里面有信息,則取出來(lái)發(fā)送到客戶端。
實(shí)現(xiàn)代碼
服務(wù)端測(cè)試代碼:server.go
package main
import (
"fmt"
"log"
"net"
"strconv"
)
func handleConnection(conn net.Conn, talkChan map[int]chan string) {
//fmt.Printf("%p\n", talkChan) //用以檢查是否是傳過(guò)來(lái)的指針
/*
定義當(dāng)前用戶的uid
*/
var curUid int
var err error
/*
定義關(guān)閉通道
*/
var closed = make(chan bool)
defer func() {
fmt.Println("defer do : conn closed")
conn.Close()
fmt.Printf("delete userid [%v] from talkChan", curUid)
delete(talkChan, curUid)
}()
/**
* 提示用戶設(shè)置自己的uid, 如果沒(méi)設(shè)置,則不朝下執(zhí)行
*/
for {
//提示客戶端設(shè)置用戶id
_, err = conn.Write([]byte("請(qǐng)?jiān)O(shè)置用戶uid"))
if err != nil {
return
}
data := make([]byte, 1024)
c, err := conn.Read(data)
if err != nil {
//closed <- true //這樣會(huì)阻塞 | 后面取closed的for循環(huán),沒(méi)有執(zhí)行到。
return
}
sUid := string(data[0:c])
//轉(zhuǎn)成int類(lèi)型
uid, _ := strconv.Atoi(sUid)
if uid < 1 {
continue
}
curUid = uid
talkChan[uid] = make(chan string)
//fmt.Println(conn, "have set uid ", uid, "can talk")
_, err = conn.Write([]byte("have set uid "+sUid+" can talk"))
if err != nil {
return
}
break
}
fmt.Println("err 3")
//當(dāng)前所有的連接
fmt.Println(talkChan)
//讀取客戶端傳過(guò)來(lái)的數(shù)據(jù)
go func() {
for {
//不停的讀客戶端傳過(guò)來(lái)的數(shù)據(jù)
data := make([]byte, 1024)
c, err := conn.Read(data)
if err != nil {
fmt.Println("have no client write", err)
closed <- true //這里可以使用 | 因?yàn)槭怯糜玫膅o 新開(kāi)的線程去處理的。 | 即便chan阻塞,后面的也會(huì)執(zhí)行去讀 closed 這個(gè)chan
}
clientString := string(data[0:c])
//將客戶端過(guò)來(lái)的數(shù)據(jù),寫(xiě)到相應(yīng)的chan里
if curUid == 3 {
talkChan[4] <- clientString
} else {
talkChan[3] <- clientString
}
}
}()
/*
從chan 里讀出給這個(gè)客戶端的數(shù)據(jù) 然后寫(xiě)到該客戶端里
*/
go func() {
for {
talkString := <-talkChan[curUid]
_, err = conn.Write([]byte(talkString))
if err != nil {
closed <- true
}
}
}()
/*
檢查是否已經(jīng)關(guān)閉連接 如果關(guān)閉則推出該線程 去執(zhí)行defer語(yǔ)句
*/
for {
if <-closed {
return
}
}
}
func main() {
/**
建立監(jiān)聽(tīng)鏈接
*/
ln, err := net.Listen("tcp", "127.0.0.1:6010")
if err != nil {
panic(err)
}
//創(chuàng)建一個(gè)管道
//talkChan := map[f]
talkChan := make(map[int]chan string)
fmt.Printf("%p\n", talkChan)
/*
監(jiān)聽(tīng)是否有客戶端過(guò)來(lái)的連接請(qǐng)求
*/
for {
fmt.Println("wait connect...")
conn, err := ln.Accept()
if err != nil {
log.Fatal("get client connection error: ", err)
}
go handleConnection(conn, talkChan)
}
}
客戶端測(cè)試代碼:client.go
package main
import (
"fmt"
"math/rand"
"net"
)
func main() {
conn, err := net.Dial("tcp", "127.0.0.1:6010")
if err != nil {
panic(err)
}
fmt.Fprintf(conn, "hello server\n")
defer conn.Close()
go writeFromServer(conn)
for {
var talkContent string
fmt.Scanln(&talkContent)
if len(talkContent) > 0 {
_, err = conn.Write([]byte(talkContent))
if err != nil {
fmt.Println("write to server error")
return
}
}
}
}
func connect() {
conn, err := net.Dial("tcp", "127.0.0.1:6010")
if err != nil {
panic(err)
}
fmt.Fprintf(conn, "hello server\n")
defer conn.Close()
go writeFromServer(conn)
for {
var talkContent string
fmt.Scanln(&talkContent)
if len(talkContent) > 0 {
_, err = conn.Write([]byte(talkContent))
if err != nil {
fmt.Println("write to server error")
return
}
}
}
}
func writeFromServer(conn net.Conn) {
defer conn.Close()
for {
data := make([]byte, 1024)
c, err := conn.Read(data)
if err != nil {
fmt.Println("rand", rand.Intn(10), "have no server write", err)
return
}
fmt.Println(string(data[0:c]) + "\n ")
}
}
相關(guān)文章
Golang創(chuàng)建構(gòu)造函數(shù)的方法超詳細(xì)講解
構(gòu)造器一般面向?qū)ο笳Z(yǔ)言的典型特性,用于初始化變量。Go語(yǔ)言沒(méi)有任何具體構(gòu)造器,但我們能使用該特性去初始化變量。本文介紹不同類(lèi)型構(gòu)造器的差異及其應(yīng)用場(chǎng)景2023-01-01golang 實(shí)現(xiàn)每隔幾分鐘執(zhí)行一個(gè)函數(shù)
這篇文章主要介紹了golang 實(shí)現(xiàn)每隔幾分鐘執(zhí)行一個(gè)函數(shù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12go語(yǔ)言題解LeetCode989數(shù)組形式的整數(shù)加法
這篇文章主要為大家介紹了go語(yǔ)言題解LeetCode989數(shù)組形式的整數(shù)加法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12Go語(yǔ)言使用組合的方式實(shí)現(xiàn)多繼承的方法
這篇文章主要介紹了Go語(yǔ)言使用組合的方式實(shí)現(xiàn)多繼承的方法,實(shí)例分析了多繼承的原理與使用組合方式來(lái)實(shí)現(xiàn)多繼承的技巧,需要的朋友可以參考下2015-02-02一文幫你搞懂Go面試中常問(wèn)的channel問(wèn)題
channel是Golang面試時(shí)經(jīng)常會(huì)問(wèn)到的問(wèn)題,所以這篇文章為大家整理了channel??嫉囊恍﹩?wèn)題以及回答,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-06-06基于Go?goroutine實(shí)現(xiàn)一個(gè)簡(jiǎn)單的聊天服務(wù)
對(duì)于聊天服務(wù),想必大家都不會(huì)陌生,因?yàn)樵谖覀兊纳钪薪?jīng)常會(huì)用到,本文我們用?Go?并發(fā)來(lái)實(shí)現(xiàn)一個(gè)聊天服務(wù)器,這個(gè)程序可以讓一些用戶通過(guò)服務(wù)器向其它所有用戶廣播文本消息,文中通過(guò)代碼示例介紹的非常詳細(xì),需要的朋友可以參考下2023-06-06Golang中crypto/rand庫(kù)的使用技巧與最佳實(shí)踐
在Golang的眾多隨機(jī)數(shù)生成庫(kù)中,crypto/rand?是一個(gè)專(zhuān)為加密安全設(shè)計(jì)的庫(kù),本文主要介紹了Golang中crypto/rand庫(kù)的使用技巧與最佳實(shí)踐,感興趣的可以了解一下2024-02-02安裝GoLang環(huán)境和開(kāi)發(fā)工具的圖文教程
Go是一門(mén)由Google開(kāi)發(fā)的編程語(yǔ)言,GoLand的安裝非常簡(jiǎn)單,本文主要介紹了安裝GoLang環(huán)境和開(kāi)發(fā)工具的圖文教程,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09