您好,登錄后才能下訂單哦!
前言
運用go里面的net包中的相關方法來實現一個基于tcp的簡單多人聊天室,用一個服務器來管理,主要反饋客戶端是否連接成功并顯示客戶端輸入的內容,并且發送給每一個在服務器上連接的客服端,下面話不多說了,來一起看看詳細的介紹吧。
示例代碼
服務器代碼
// server package main import ( "fmt" "net" ) var ConnMap map[string]*net.TCPConn func checkErr(err error) int { if err != nil { if err.Error() == "EOF" { fmt.Println("用戶退出") return 0 } fmt.Println("發生錯誤") return -1 } return 1 } func say(tcpConn *net.TCPConn) { for { data := make([]byte, 256) total, err := tcpConn.Read(data) if err != nil { fmt.Println(string(data[:total]), err) } else { fmt.Println(string(data[:total])) } flag := checkErr(err) if flag == 0 { break } for _, conn := range ConnMap { if conn.RemoteAddr().String() == tcpConn.RemoteAddr().String() { continue } conn.Write(data[:total]) } } } func main() { //var conn net.TCPConn //localAddr :=conn.LocalAddr().String() //fmt.Println(localAddr) //tcpAddr, _ := net.ResolveTCPAddr("tcp",localAddr) tcpAddr, _ := net.ResolveTCPAddr("tcp", "192.168.128.216:8080") tcpListen, _ := net.ListenTCP("tcp", tcpAddr) ConnMap = make(map[string]*net.TCPConn) for { tcpConn, _ := tcpListen.AcceptTCP() defer tcpConn.Close() ConnMap[tcpConn.RemoteAddr().String()] = tcpConn fmt.Println("連接客戶端信息:", tcpConn.RemoteAddr().String()) go say(tcpConn) } }
本來打算用系統的LocalAddr()
函數來直接調用本地網絡地址,但是這里調用一直報格式的錯誤,所以這里用本地地址直接輸在里面,端口是8080//192.168.247.128:8080,也可以改成和客戶端相似的代碼來進行調用,但鑒于服務器唯一,所以就不做更改了
客戶端代碼
// client package main import ( "fmt" "net" "os" ) var ch chan int = make(chan int) var nickname string func reader(conn *net.TCPConn) { buff := make([]byte, 256) for { j, err := conn.Read(buff) if err != nil { ch <- 1 break } fmt.Printf("%s\n", buff[0:j]) } } func main() { if len(os.Args) != 2 { fmt.Fprintf(os.Stderr, "Usage:%s host:port", os.Args[0]) os.Exit(1) } service := os.Args[1] TcpAdd, _ := net.ResolveTCPAddr("tcp", service) //TcpAdd, _ := net.ResolveTCPAddr("tcp", "localhost:8080") conn, err := net.DialTCP("tcp", nil, TcpAdd) if err != nil { fmt.Println("服務沒打開") os.Exit(1) } defer conn.Close() go reader(conn) fmt.Println("請輸入昵稱") fmt.Scanln(&nickname) fmt.Println("你的昵稱是:", nickname) //var str string for { var msg string fmt.Scan(&msg) fmt.Print("<" + nickname + ">" + "說:") //for i, _ := range msg { // fmt.Printf("%c", msg[i]) //} fmt.Println(msg) b := []byte("<" + nickname + ">" + "說:" + msg) conn.Write(b) select { case <-ch: fmt.Println("server發生錯誤,請重新連接") os.Exit(2) default: } } }
客戶端主要就是向服務器發起連接,然后發送和接收數據,原計劃定義一個字符切片來存儲讀來的數據,但是出現了調用fmt.Println()來輸出是時候顯示ASCII碼的錯誤,所以改成直接往一個空字符串里面讀寫,然后再來調用fmt.Println()
,顯示出正確的字符串
運行服務器
go build server.go
./server
或 go run server.go
運行客戶端
go build client.go
./client 服務器地址:端口(上面例子是192.168.247.128:8080)
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對億速云的支持。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。