• 设为首页
  • 收藏本站
  • 积分充值
  • VIP赞助
  • 手机版
  • 微博
  • 微信
    微信公众号 添加方式:
    1:搜索微信号(888888
    2:扫描左侧二维码
  • 快捷导航
    福建二哥 门户 查看主题

    基于Go语言实现简单网络聊天室(命令行模式)

    发布者: 琛瑞6678 | 发布时间: 2025-8-14 05:42| 查看数: 98| 评论数: 0|帖子模式

    实战简介

    网络聊天室(命令行模式)
    要求:

    • 输入网名,可以进入聊天室
    • 聊天内信息实时更新
    • 利用协程处理多任务并发
    基于tcp协议实现功能
    服务器端
    接受用户消息和循环转发
    对功能命令进行处理

    • ./cd1 或 ./menu 功能菜单
    • ./cd2 或 ./changeName 更改昵称
    • ./cd3 或 ./online 在线用户数量查询
    • ./cd4 或 ./quit 退出聊天室
    客户端
    接受服务器发送的信息并处理
    接受用户的输入处理后发往服务器

    结构和示例



    用户登录示例



    功能行命令测试



    发送消息广播测试



    用户退出示例



    服务器端



    基本流程

    1. 初始化
    init() 函数被调用,用于初始化全局变量:

    • 创建一个映射 onlineList 用来保存在线用户的信息。
    • 创建一个带缓冲的消息通道 message 用于广播消息给所有在线用户。
    2. 主函数 main()

    • 使用 net.Listen 监听 TCP 地址 "127.0.0.1:8080"。
    • 启动 manger() 协程来监听消息通道 message,并将消息广播给所有在线用户。
    • 主循环中,使用 Accept 接受新的客户端连接,并为每个客户端启动一个新的协程 handleConnection(conn)。
    3. 处理客户端连接 handleConnection(conn)
    对于每个客户端连接,首先增加在线用户计数 count。
    调用 addUser(conn) 添加新用户到在线用户列表,并返回一个 client 结构体实例。
    创建一个 quit 通道,用于在客户端断开连接时发送信号。
    启动两个协程:

    • writeMsgToClient(conn, quit):从用户的 userChannel 中读取消息并发送给客户端。
    • readClient(conn, quit):从客户端读取消息并处理。
    监听 quit 通道以检测客户端是否断开连接。
    4. 添加新用户 addUser(conn)
    创建一个新的 client 实例,其中包含一个用于消息的通道 userChannel、客户端连接 conn 和默认名称(客户端地址)。
    将新用户添加到 onlineList 映射中。
    5. 管理消息广播 manger()
    从消息通道 message 中读取消息,并将消息广播给所有在线用户。
    6. 写消息到客户端 writeMsgToClient(conn, quit)
    从客户端的 userChannel 读取消息,使用 module.Encode 对消息进行编码,并通过客户端连接 conn 发送给客户端。
    如果发生错误或客户端断开连接,关闭客户端连接并发送信号到 quit 通道。
    7. 读取客户端消息 readClient(conn, quit)
    从客户端读取消息,根据消息的内容执行不同的操作:

    • 如果消息是以 !@#$@!cd1changeName 开头,则处理昵称更改请求。
    • 如果消息是以 !@#$@!cd4exit 开头,则处理客户端退出请求。
    • 如果消息是以 !@#$@!menu 开头,则向客户端发送菜单命令。
    • 如果消息是以 !@#$@!cd3online 开头,则向客户端发送在线用户数量。
    • 其他情况下,将消息广播给所有在线用户。
    如果客户端断开连接,则发送信号到 quit 通道。
    8. 处理客户端退出
    当 quit 通道接收到信号时,从在线用户列表中删除该客户端,并减少在线用户计数。
    如果所有客户端都已断开连接,则输出“等待用户连接中…”。


    代码
    1. package main

    2. import (
    3.         "bufio"         
    4.         "chatRoom/chatRoom/module" // 消息的编码和解码模块
    5.         "fmt"         
    6.         "io"            
    7.         "log"            
    8.         "net"            
    9.         "strconv"        
    10.         "strings"      
    11.         "sync"           
    12.         "time"         
    13. )

    14. // 定义客户端结构体
    15. type client struct {
    16.         userChannel chan string // 用户的消息通道
    17.         conn        net.Conn    // 网络连接
    18.         name        string      // 用户名
    19.         addr        string      // 客户端地址
    20. }

    21. // 定义在线用户计数器
    22. var count int

    23. // 定义互斥锁
    24. var mu sync.Mutex

    25. // 定义在线用户列表
    26. var onlineList map[string]*client

    27. // 定义消息广播通道
    28. var message chan string

    29. // 初始化函数
    30. func init() {
    31.         onlineList = make(map[string]*client) // 初始化在线用户列表
    32.         message = make(chan string, 1024)     // 初始化消息广播通道
    33. }

    34. // 主函数
    35. func main() {
    36.         fmt.Println("端口监听中...")
    37.         listener, err := net.Listen("tcp", "127.0.0.1:8080")
    38.         if err != nil {
    39.                 log.Fatal(err) // 如果监听失败,记录错误并退出程序
    40.         }
    41.         defer listener.Close()
    42.         time.Sleep(time.Second)
    43.         fmt.Println("端口监听成功")

    44.         // 启动管理消息广播的协程
    45.         go manger()

    46.         // 主循环,接受客户端连接
    47.         for {
    48.                 conn, err := listener.Accept()
    49.                 if err != nil {
    50.                         continue
    51.                 }
    52.                 go handleConnection(conn)
    53.         }
    54. }

    55. // 处理客户端连接的函数
    56. func handleConnection(conn net.Conn) {
    57.         defer conn.Close()

    58.         count++ // 增加在线用户计数

    59.         fmt.Println("有新用户连接服务,当前连接数:", count)

    60.         // 添加新用户
    61.         addUser(conn)

    62.         // 创建退出信号通道
    63.         var quit = make(chan bool)

    64.         // 启动写消息到客户端的协程
    65.         go writeMsgToClient(conn, quit)

    66.         // 启动读取客户端消息的协程
    67.         go readClient(conn, quit)

    68.         // 监听退出信号
    69.         select {
    70.         case <-quit:
    71.                 // 用户下线处理
    72.                 connName := onlineList[conn.RemoteAddr().String()].name
    73.                 mu.Lock()
    74.                 close(onlineList[conn.RemoteAddr().String()].userChannel)
    75.                 mu.Unlock()
    76.                 mu.Lock()
    77.                 delete(onlineList, conn.RemoteAddr().String())
    78.                 mu.Unlock()
    79.                 count--
    80.                 message <- "< 系统消息 > [ " + connName + " ]" + "下线了 当前在线人数 " + strconv.Itoa(len(onlineList)) + " 人"
    81.                 fmt.Println("有用户下线了,当前连接数:", count)
    82.                 if count == 0 {
    83.                         fmt.Println("等待用户连接中...")
    84.                 }

    85.                 return
    86.         }
    87. }

    88. // 修改用户名的方法
    89. func (c *client) changeName(newUserName string) bool {
    90.         mu.Lock()
    91.         defer mu.Unlock()
    92.         // 更新用户名
    93.         c.name = newUserName
    94.         return true
    95. }

    96. // 管理消息广播的函数
    97. func manger() {
    98.         fmt.Println("开始监听 message通道")
    99.         defer fmt.Println("结束监听 message通道")
    100.         for msg := range message {
    101.                 mu.Lock()
    102.                 for _, v := range onlineList {
    103.                         v.userChannel <- msg
    104.                 }
    105.                 mu.Unlock()
    106.         }
    107. }

    108. // 写消息到客户端的协程
    109. func writeMsgToClient(conn net.Conn, quit chan bool) {
    110.         fmt.Println(onlineList[conn.RemoteAddr().String()].name, "的信息通道监听成功")
    111.         defer fmt.Println(onlineList[conn.RemoteAddr().String()].name, "的信息通道监听结束")
    112.         for msg := range onlineList[conn.RemoteAddr().String()].userChannel {
    113.                 king, err := module.Encode(msg + "\n")
    114.                 if err != nil {
    115.                         fmt.Println("发送消息失败")
    116.                         continue
    117.                 }
    118.                 _, err = conn.Write(king)
    119.                 if err != nil {
    120.                         fmt.Println("发送消息失败")
    121.                         quit <- true
    122.                 }
    123.         }
    124.         fmt.Println("函数writeMsgToClient函数结束")
    125. }

    126. // 添加新用户
    127. func addUser(conn net.Conn) client {
    128.         fmt.Println("开始使用添加新用户" + conn.RemoteAddr().String())
    129.         newUser := client{
    130.                 make(chan string), // 创建用户消息通道
    131.                 conn,              // 网络连接
    132.                 conn.RemoteAddr().String(), // 用户名,初始化为客户端地址
    133.                 conn.RemoteAddr().String(), // 客户端地址
    134.         }
    135.         onlineList[conn.RemoteAddr().String()] = &newUser // 添加到在线用户列表

    136.         fmt.Println("addUser函数结束,用户" + conn.RemoteAddr().String() + "添加成功")
    137.         return newUser
    138. }

    139. // 读取客户端消息的协程
    140. func readClient(conn net.Conn, quit chan bool) {
    141.         fmt.Println("开始读取客户端发送的信息")
    142.         defer fmt.Println("客户端发送信息读取结束")

    143.         userChannel := onlineList[conn.RemoteAddr().String()].userChannel

    144.         reader := bufio.NewReader(conn)
    145.         for {
    146.                 msg, err := module.Decode(reader)
    147.                 if err == io.EOF {
    148.                         quit <- true
    149.                 }
    150.                 if err != nil {
    151.                         fmt.Println("decode msg failed, err:", err)
    152.                         quit <- true
    153.                 }
    154.                 if len(msg) == 0 {
    155.                         continue
    156.                 }
    157.                 fmt.Println("收到client发来的数据:", msg)

    158.                 // 处理客户端发送的不同类型的消息
    159.                 switch {
    160.                 case strings.HasPrefix(msg, "!@#$@!cd1changeName"):

    161.                         king := true
    162.                         oldName := onlineList[conn.RemoteAddr().String()].name
    163.                         newName := strings.TrimPrefix(msg, "!@#$@!cd1changeName")

    164.                         if strings.HasPrefix(msg, "!@#$@!cd1changeNameFirst") {
    165.                                 newName = strings.TrimPrefix(msg, "!@#$@!cd1changeNameFirst")
    166.                         }

    167.                         if newName == "" {
    168.                                 newName = conn.RemoteAddr().String()
    169.                         }

    170.                         for _, v := range onlineList {

    171.                                 mapName := v.name
    172.                                 if mapName == newName {

    173.                                         king = false

    174.                                         break
    175.                                 }
    176.                         }

    177.                         if strings.HasPrefix(msg, "!@#$@!cd1changeNameFirst") && king == false {

    178.                                 message <- "< 系统消息 > [ " + conn.RemoteAddr().String() + " ] [ " + oldName + " ] 上线了!"

    179.                                 userChannel <- "< 系统消息 > [ " + onlineList[conn.RemoteAddr().String()].name + " ]" + "名字: " + newName + " 已存在,请更换一个名字尝试"

    180.                                 userChannel <- "< 系统消息 > 你当前昵称为: " + oldName + " ( 输入cd2可进行名字修改 )"

    181.                                 continue
    182.                         }

    183.                         if king == false {

    184.                                 userChannel <- "< 系统消息 > 昵称修改失败!!!"

    185.                                 userChannel <- "< 系统消息 > [ " + onlineList[conn.RemoteAddr().String()].name + " ]" + "名字: " + newName + " 已存在,请更换一个名字尝试"

    186.                                 userChannel <- "< 系统消息 > 你当前昵称为: " + oldName

    187.                                 continue

    188.                         }

    189.                         isSuccess := onlineList[conn.RemoteAddr().String()].changeName(newName)
    190.                         if isSuccess {

    191.                                 userChannel <- "!@#$@!cd1changeName" + newName

    192.                                 if strings.HasPrefix(msg, "!@#$@!cd1changeNameFirst") {

    193.                                         message <- "< 系统消息 > [ " + conn.RemoteAddr().String() + " ] [ " + newName + " ] 上线了!"

    194.                                         time.Sleep(time.Millisecond * 50)

    195.                                         userChannel <- "< 系统消息 > 你当前的昵称为:" + newName

    196.                                         continue
    197.                                 }

    198.                                 userChannel <- "< 系统消息 > 昵称修改成功 你当前昵称为: " + newName

    199.                         } else {

    200.                                 userChannel <- "< 系统消息 > 昵称修改失败!!!"

    201.                         }

    202.                         message <- "< 系统消息 > [ " + conn.RemoteAddr().String() + " ]" + " 旧昵称为: " + oldName + " 新昵称为: " + newName

    203.                 case strings.HasPrefix(msg, "!@#$@!cd4exit"):
    204.                         fmt.Println("[ " + onlineList[conn.RemoteAddr().String()].name + " ] " + "下线了")

    205.                         quit <- true

    206.                         return
    207.                 case strings.HasPrefix(msg, "!@#$@!menu"):

    208.                         userChannel <- "< 系统消息 > \n * ./cd1 或 ./menu       功能菜单\n * ./cd2 或 ./changeName 更改昵称\n * ./cd3 或 ./online     在线用户数量查询\n * ../cd4 或 ./quit      退出聊天室"

    209.                 case strings.HasPrefix(msg, "!@#$@!cd3online"):

    210.                         fmt.Println("在线人数:", count)

    211.                         userChannel <- "< 系统消息 > 当前在线人数:" + strconv.Itoa(count)

    212.                 default:

    213.                         message <- "[ " + onlineList[conn.RemoteAddr().String()].name + " ]" + ": " + msg

    214.                         fmt.Println("信息广播成功")
    215.                 }
    216.         }
    217. }
    复制代码
    客户端

    基本流程

    1. 主函数 main()

    • 尝试连接到服务器 "127.0.0.1:8080"。
    • 如果连接失败,打印错误信息并退出程序。
    • 如果连接成功,获取用户的昵称并发送给服务器。
    • 创建一个 exit 通道,用于接收退出信号。
    • 打印欢迎信息和命令提示。
    • 启动一个协程 readMsg(conn) 用于读取消息。
    • 启动另一个协程用于处理用户输入。
    • 主循环中监听 exit 通道,如果接收到信号则退出程序。
    2. 获取用户输入 getUserInput(prompt string)

    • 根据传入的提示信息显示相应的提示。
    • 读取用户从标准输入的输入。
    • 返回去除空白字符的输入字符串。
    3. 读取消息 readMsg(conn)

    • 从服务器读取消息。
    • 如果读取到 EOF (文件结束),则表示服务器连接已断开,终止程序。
    • 如果读取过程中出现其他错误,则打印错误信息并退出函数。
    • 如果接收到的消息为空,则跳过本次循环。
    • 如果接收到的消息是以 !@#$@!cd1changeName 开头,则更新用户的昵称。
    • 打印接收到的消息的时间戳和内容。
    4. 处理用户输入

    • 循环读取用户输入,并根据不同的命令构建消息。
    • 如果命令是 ./cd1 或 ./menu,则发送 !@#$@!menu 消息。
    • 如果命令是 ./cd2 或 ./changeName,则请求用户输入新的昵称,并发送 !@#$@!cd1changeName 加上新的昵称。
    • 如果命令是 ./cd3 或 ./online,则发送 !@#$@!cd3online 消息。
    • 如果命令是 cd4 或 ./quit,则发送 !@#$@!cd4exit 消息,并发送退出信号到 exit 通道。
    • 对于其他消息,编码并发送到服务器。
    代码
    1. package main

    2. import (
    3.         "bufio"         
    4.         "chatRoom/chatRoom/module" // 消息的编码和解码模块
    5.         "fmt"            
    6.         "io"            
    7.         "net"            
    8.         "os"            
    9.         "strings"        
    10.         "time"           
    11. )

    12. // 定义一个全局变量用于存储用户的昵称
    13. var name string

    14. // 主函数
    15. func main() {
    16.         // 尝试连接到服务器
    17.         conn, err := net.Dial("tcp", "127.0.0.1:8080")
    18.         if err != nil {
    19.                 fmt.Println("服务器连接失败 err =", err)
    20.                 return
    21.         }

    22.         defer conn.Close()
    23.         // 打印连接成功的消息
    24.         fmt.Println("服务器连接成功")

    25.         // 获取用户的昵称
    26.         name = getUserInput("请输入你的昵称:")
    27.         // 构建一条特殊的消息,用于通知服务器用户昵称
    28.         data, err := module.Encode("!@#$@!cd1changeNameFirst" + name)
    29.         if err != nil {
    30.                
    31.                 fmt.Println("encode msg failed, err:", err)
    32.                 return
    33.         }
    34.         // 发送消息到服务器
    35.         _, err = conn.Write(data)
    36.         if err != nil {
    37.                 fmt.Println("发送数据失败1 err =", err)
    38.         }

    39.         // 创建一个通道,用于接收退出信号
    40.         var exit = make(chan bool)
    41.         // 确保在函数退出时关闭通道
    42.         defer close(exit)

    43.         // 显示欢迎信息和命令提示
    44.         fmt.Println("--------------欢迎进入多人聊天室系统----------------")
    45.         fmt.Println("       * ./cd1 或 ./menu       功能菜单")
    46.         fmt.Println("       * ./cd2 或 ./changeName 更改昵称")
    47.         fmt.Println("       * ./cd3 或 ./online     在线用户数量查询")
    48.         fmt.Println("       * ./cd4 或 ./quit       退出聊天室")
    49.         fmt.Println("---------------指令字母不区分大小写-----------------")

    50.         // 启动一个协程用于读取消息
    51.         go readMsg(conn)

    52.         // 启动一个协程用于处理用户输入
    53.         go func() {
    54.                 for {
    55.                         // 获取用户输入
    56.                         msg := getUserInput("")

    57.                         // 根据用户输入特殊消息处理命令
    58.                         if strings.EqualFold(msg, "./cd1") || strings.EqualFold(msg, "./menu") {
    59.                                 msg = "!@#$@!menu"
    60.                         }
    61.                         if strings.EqualFold(msg, "./cd2") || strings.EqualFold(msg, "./changeName") {
    62.                                 newMsg := getUserInput("请输入新的昵称:")
    63.                                 msg = "!@#$@!cd1changeName" + newMsg
    64.                         }
    65.                         if strings.EqualFold(msg, "./cd3") || strings.EqualFold(msg, "./online") {
    66.                                 msg = "!@#$@!cd3online"
    67.                         }
    68.                         if strings.EqualFold(msg, "./cd4") || strings.EqualFold(msg, "./quit") {
    69.                                 msg = "!@#$@!cd4exit"
    70.                                 // 编码并发送退出消息
    71.                                 data, err := module.Encode(msg)
    72.                                 if err != nil {
    73.                                         fmt.Println("消息数据失败1, err:", err)
    74.                                         return
    75.                                 }
    76.                                 _, err = conn.Write(data)
    77.                                 if err != nil {
    78.                                         // 如果发送失败,打印错误信息
    79.                                         fmt.Println("发送数据失败2 err =", err)
    80.                                 }
    81.                                 // 打印退出信息
    82.                                 fmt.Println("正在退出...")
    83.                                 // 发送退出信号
    84.                                 exit <- true
    85.                                 return
    86.                         }

    87.                         // 编码并发送普通消息
    88.                         data, err := module.Encode(msg)
    89.                         if err != nil {
    90.                                 // 如果消息编码失败,打印错误信息并退出协程
    91.                                 fmt.Println("发送数据失败3, err:", err)
    92.                                 return
    93.                         }
    94.                         _, err = conn.Write(data)
    95.                         if err != nil {
    96.                                 // 如果发送失败,打印错误信息
    97.                                 fmt.Println("发送数据失败4 err =", err)
    98.                         }
    99.                 }
    100.         }()

    101.         // 主循环,监听退出信号
    102.         for {
    103.                 select {
    104.                 case <-exit:
    105.                         // 当收到退出信号时,打印退出成功并退出程序
    106.                         fmt.Println("退出成功")
    107.                         return
    108.                 }
    109.         }
    110. }

    111. // getUserInput 函数用于获取用户输入
    112. func getUserInput(prompt string) string {

    113.         time.Sleep(time.Millisecond * 100)
    114.         // 根据不同的提示信息显示相应的提示
    115.         switch prompt {
    116.         case "请输入你的昵称:":
    117.                 fmt.Print("请输入你的昵称:")
    118.         case "请输入新的昵称:":
    119.                 fmt.Println("请输入新的昵称:")
    120.         }
    121.         // 创建一个标准输入的缓冲读取器
    122.         reader := bufio.NewReader(os.Stdin)
    123.         // 读取一行输入
    124.         input, err := reader.ReadString('\n')
    125.         if err != nil {
    126.                 // 如果读取失败,打印错误信息并返回错误信息
    127.                 fmt.Println("用户输入获取失败:err =", err)
    128.                 return "客户端信息读取错误"
    129.         }
    130.         // 返回去掉空格的输入字符串
    131.         return strings.TrimSpace(input)
    132. }

    133. // readMsg 函数用于读取消息
    134. func readMsg(conn net.Conn) {

    135.         defer conn.Close()
    136.         // 创建一个缓冲读取器来读取连接中的数据
    137.         reader := bufio.NewReader(conn)
    138.         for {
    139.                 // 解码消息
    140.                 msg, err := module.Decode(reader)
    141.                 if err == io.EOF {
    142.                         // 如果遇到EOF(文件结束),表示连接已断开
    143.                         fmt.Println("服务器连接已断开 ")
    144.                         // 终止程序
    145.                         os.Exit(1)
    146.                 }
    147.                 if err != nil {
    148.                
    149.                         fmt.Println("服务器断开连接 2 err =", err)
    150.                         return
    151.                 }
    152.                 if msg == "" {
    153.                         // 如果消息为空,则跳过本次循环
    154.                         continue
    155.                 }
    156.                 if strings.HasPrefix(msg, "!@#$@!cd1changeName") {
    157.                         // 如果消息是更改昵称的通知
    158.                         msg1 := strings.TrimPrefix(msg, "!@#$@!cd1changeName")
    159.                         name = strings.TrimRight(msg1, "\n")
    160.                         // 更新昵称
    161.                         continue
    162.                 }
    163.                 // 打印消息的时间戳和内容
    164.                 fmt.Print("【 ", time.Now().Format("15:04"), " 】", msg)
    165.         }
    166. }
    复制代码
    消息封包和解包的函数

    作用:防止tcp粘包的情况影响消息的读取
    1.为什么会出现粘包
    主要原因就是tcp数据传递模式是流模式,在保持长连接的时候可以进行多次的收和发。
    “粘包"可发生在发送端也可发生在接收端:

    • 由Nagle算法造成的发送端的粘包:Nagle算法是一种改善网络传输效率的算法。简单来说就是当我们提交一段数据给TCP发送时,TCP并不立刻发送此段数据,而是等待一小段时间看看在等待期间是否还有要发送的数据,若有则会一次把这两段数据发送出去。
    • 接收端接收不及时造成的接收端粘包:TCP会把接收到的数据存在自己的缓冲区中,然后通知应用层取数据。当应用层由于某些原因不能及时的把TCP的数据取出来,就会造成TCP缓冲区中存放了几段数据。
    2.解决办法
    出现"粘包"的关键在于接收方不确定将要传输的数据包的大小,因此我们可以对数据包进行封包和拆包的操作。
    封包:封包就是给一段数据加上包头,这样一来数据包就分为包头和包体两部分内容了(过滤非法包时封包会加入"包尾"内容)。包头部分的长度是固定的,并且它存储了包体的长度,根据包头长度固定以及包头中含有包体长度的变量就能正确的拆分出一个完整的数据包。
    我们可以自己定义一个协议,比如数据包的前4个字节为包头,里面存储的是发送的数据的长度。
    代码
    1. package module

    2. import (
    3.         "bufio"
    4.         "bytes"
    5.         "encoding/binary"
    6. )

    7. func Encode(message string) ([]byte, error) {
    8.         // 读取消息的长度,转换成int32类型(占4个字节)
    9.         var length = int32(len(message))
    10.         var pkg = new(bytes.Buffer)
    11.         // 写入消息头
    12.         err := binary.Write(pkg, binary.LittleEndian, length)
    13.         if err != nil {
    14.                 return nil, err
    15.         }
    16.         // 写入消息实体
    17.         err = binary.Write(pkg, binary.LittleEndian, []byte(message))
    18.         if err != nil {
    19.                 return nil, err
    20.         }
    21.         return pkg.Bytes(), nil
    22. }

    23. // Decode 解码消息
    24. func Decode(reader *bufio.Reader) (string, error) {
    25.         // 读取消息的长度
    26.         lengthByte, _ := reader.Peek(4) // 读取前4个字节的数据
    27.         lengthBuff := bytes.NewBuffer(lengthByte)
    28.         var length int32
    29.         err := binary.Read(lengthBuff, binary.LittleEndian, &length)
    30.         if err != nil {
    31.                 return "", err
    32.         }
    33.         // Buffered返回缓冲中现有的可读取的字节数。
    34.         if int32(reader.Buffered()) < length+4 {
    35.                 return "", err
    36.         }

    37.         // 读取真正的消息数据
    38.         pack := make([]byte, int(4+length))
    39.         _, err = reader.Read(pack)
    40.         if err != nil {
    41.                 return "", err
    42.         }
    43.         return string(pack[4:]), nil
    44. }
    复制代码
    以上就是基于Go语言实现简单网络聊天室(命令行模式)的详细内容,更多关于Go聊天室的资料请关注脚本之家其它相关文章!

    来源:互联网
    免责声明:如果侵犯了您的权益,请联系站长(1277306191@qq.com),我们会及时删除侵权内容,谢谢合作!

    本帖子中包含更多资源

    您需要 登录 才可以下载或查看,没有账号?立即注册

    ×

    最新评论

    QQ Archiver 手机版 小黑屋 福建二哥 ( 闽ICP备2022004717号|闽公网安备35052402000345号 )

    Powered by Discuz! X3.5 © 2001-2023

    快速回复 返回顶部 返回列表