美文网首页PHP高级技能
go写的简易聊天

go写的简易聊天

作者: 学不会swoole不改名 | 来源:发表于2019-05-13 20:07 被阅读25次

    学习go的小进步

    server 代码
    //chat server listen 8080
    package main
    
    import (
        "fmt"
        "log"
        "net"
        "os"
        "strings"
    )
    
    const (
        LOG_DIRECTORY = "./test.log" //记录错误日志的路径
    )
    
    var online = make(map[string]net.Conn)
    var msgQyeue = make(chan string, 1000)
    var quitChan = make(chan bool)
    var logFile *os.File
    var logger *log.Logger
    
    func coninfo(con net.Conn) {
        //make a buff 消息的缓冲区
        buff := make([]byte, 1024)
        //最后记得关掉 将退出连接的请求删掉
        defer func(con net.Conn) {
            addr := fmt.Sprintf("%s", con.RemoteAddr())
            delete(online, addr)
            con.Close()
        }(con)
        //不断的读取
        for {
            numBuff, err := con.Read(buff)
            if err != nil {
                continue
            }
            if numBuff != 0 {
                msgQyeue <- string(buff[:numBuff]) //细节 因为我们最大1024字节,会有上一条消息的缓存 生产者
            }
        }
    }
    
    //消费数据
    func outMsg() {
        for {
            select {
            case msg := <-msgQyeue:
                //doMsg(<-msgQyeue) 这样写client接收不到返回的消息
                doMsg(msg)
            case <-quitChan:
                break
    
            }
        }
    }
    
    //对数据做处理 127.0.0.0:1234#你好
    func doMsg(msg string) {
        content := strings.Split(msg, "#") //按照#切分数据
        if len(content) > 1 {
            addr := content[0]
            sendMsg := strings.Join(content[1:], "#") //127.0.0.1 #(你好#xxx)->连接起来
            addr = strings.Trim(addr, " ")            //对空格进行处理相当于phptrim
            if con, ok := online[addr]; ok {
                _, err := con.Write([]byte(sendMsg))
                if err != nil {
                    fmt.Println("send msg is nil")
                }
    
            }
        } else {
            content := strings.Split(msg, "*") //自定义规则按照*切分数据
            if strings.ToUpper(content[1]) == "LIST" {
                var ips = ""
                for i := range online {
                    ips = ips + "|" + i
                }
                if con, ok := online[content[0]]; ok {
                    _, err := con.Write([]byte(ips))
                    if err != nil {
                        fmt.Println("send msg is nil")
                    }
                }
            }
        }
    }
    func main() {
        //打开日志文件
        logFile, err := os.OpenFile(LOG_DIRECTORY, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0)
        if nil != err {
            logger.Println(err)
            os.Exit(-1)
        }
    
        defer logFile.Close()
        logger = log.New(logFile, "\r\n", log.Ldate|log.Ltime|log.Llongfile)
    
        //server 监听的端口
        socketer, err := net.Listen("tcp", "localhost:8080")
        if err != nil {
            fmt.Println(err)
        }
        defer socketer.Close()
        fmt.Println("wait for message……")
        logger.Println("i am writing the log")
    
        go outMsg()
        for {
            //accept是建立链接后进行通信的 如果不用协程效果是你说一句我说一句
            con, err := socketer.Accept()
            if err != nil {
                fmt.Println(err)
            }
            //将con 映射到map里面
            //con.RemoteAddr() 获取cil ip和端口
            addr := fmt.Sprintf("%s", con.RemoteAddr())
            online[addr] = con
            for i := range online {
                fmt.Printf("%s\n", i)
                logger.Printf("%s\n", i)
            }
            //协程read
            go coninfo(con)
        }
    
    }
    
    
    client 代码
    //chat here is cilent
    package main
    
    import (
        "bufio"
        "strings"
    
        "fmt"
        "net"
        "os"
    )
    
    func checkErr(err error) {
        if err != nil {
            fmt.Println(err)
    
        }
    }
    
    func sendMessage(con net.Conn) {
        var input string //没什么暖用的定义 ,就是为了byte少了一层string的转换
        //cil协程write server协程read
        for {
            reader := bufio.NewReader(os.Stdin)   //bufio来读取标准输入
            data, _, _ := reader.ReadLine()       //返回你输入那一行的data(数据)
            input = string(data)                  //感觉没什么暖用,还是写上了
            if strings.ToUpper(input) == "EXIT" { //如果 输入的是EXIT就退出server,退出前关闭close这是一个好习惯
                con.Close()
                break
            }
            _, err := con.Write(data) //通过write来写数据 server通过read来读数据
            if err != nil {
                con.Close() //同上
                break
            }
        }
    }
    func main() {
    
        con, err := net.Dial("tcp", "localhost:8080") //连接server的ip和server的端口
        checkErr(err)
        defer con.Close() //好习惯 nbb
    
        go sendMessage(con) //协程写入
        // checkErr(err)
        buf := make([]byte, 1024)
        for {
            numOfBytes, err := con.Read(buf) //localhost:8080读取数据
            if err != nil {
                fmt.Println("您已经退出,感谢您的使用")
                os.Exit(0)
            }
            // if string(buf[:numOfBytes]) == "你好" {
            //  fmt.Println("recv meg is:" + string("你好大傻逼"))
            //  return
            // }
            fmt.Println("recv meg is:" + string(buf[:numOfBytes]))
        }
        fmt.Println("message is end !")
    
    }
    
    
    默认client发送需要另一个client的ip#content的格式,这个格式自定义,过几天用swoole实现以下这个,对比以下如果查看ip list 格式是ip*list这个也可以自定义

    注释很详细,逻辑很清晰,给自己一个赞

    相关文章

      网友评论

        本文标题:go写的简易聊天

        本文链接:https://www.haomeiwen.com/subject/cdtdaqtx.html