欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Golang實(shí)現(xiàn)的聊天程序服務(wù)端和客戶端代碼分享

 更新時間:2014年10月29日 13:25:32   投稿:junjie  
這篇文章主要介紹了Golang實(shí)現(xiàn)的聊天程序服務(wù)端和客戶端代碼分享,本文先是講解了實(shí)現(xiàn)邏輯,然后給出了實(shí)現(xiàn)代碼,需要的朋友可以參考下

實(shí)現(xiàn)邏輯

1、Golang 版本  1.3

2、實(shí)現(xiàn)原理:

  1、主進(jìn)程建立TCP監(jiān)聽服務(wù),并且初始化一個變量 talkChan := make(map[int]chan string)

  2、當(dāng)主進(jìn)程ACCEPT連接請求后,利用go 啟動一個協(xié)程A去維持和客戶端的連接,把taokChan帶入到協(xié)程里

  3、和客戶端建立連接的協(xié)程A,發(fā)送消息給客戶端,使其發(fā)送自己的用戶信息。

  4、協(xié)程A在收到客戶端發(fā)送的用戶信息后,建立一個此用戶對應(yīng)的管道 talkChan[uid] = make(chan string)

  5、協(xié)程A再啟動一個協(xié)程A1去專門用來讀取客戶端發(fā)送的消息,并且用來判斷是發(fā)送給誰的消息,然后把消息放到對應(yīng)的chan里。

  6、協(xié)程A再啟動一個協(xié)程A2用來讀取此用戶對應(yīng)的管道,如果里面有信息,則取出來發(fā)送到客戶端。

實(shí)現(xiàn)代碼

服務(wù)端測試代碼:server.go

復(fù)制代碼 代碼如下:

package main

import (
    "fmt"
    "log"
    "net"
    "strconv"
)

func handleConnection(conn net.Conn, talkChan map[int]chan string) {
    //fmt.Printf("%p\n", talkChan)  //用以檢查是否是傳過來的指針

    /*
        定義當(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, 如果沒設(shè)置,則不朝下執(zhí)行
     */
    for {
        //提示客戶端設(shè)置用戶id
        _, err = conn.Write([]byte("請設(shè)置用戶uid"))
        if err != nil {
            return
        }
        data := make([]byte, 1024)
        c, err := conn.Read(data)
        if err != nil {
            //closed <- true  //這樣會阻塞 | 后面取closed的for循環(huán),沒有執(zhí)行到。
            return
        }
        sUid := string(data[0:c])

        //轉(zhuǎn)成int類型
        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)

    //讀取客戶端傳過來的數(shù)據(jù)
    go func() {
        for {
            //不停的讀客戶端傳過來的數(shù)據(jù)
            data := make([]byte, 1024)
            c, err := conn.Read(data)
            if err != nil {
                fmt.Println("have no client write", err)
                closed <- true //這里可以使用 | 因為是用用的go 新開的線程去處理的。 |  即便chan阻塞,后面的也會執(zhí)行去讀 closed 這個chan
            }

            clientString := string(data[0:c])

            //將客戶端過來的數(shù)據(jù),寫到相應(yīng)的chan里
            if curUid == 3 {
                talkChan[4] <- clientString
            } else {
                talkChan[3] <- clientString
            }

        }
    }()

    /*
        從chan 里讀出給這個客戶端的數(shù)據(jù) 然后寫到該客戶端里
    */
    go func() {
        for {
            talkString := <-talkChan[curUid]
            _, err = conn.Write([]byte(talkString))
            if err != nil {
                closed <- true
            }
        }
    }()

    /*
       檢查是否已經(jīng)關(guān)閉連接 如果關(guān)閉則推出該線程  去執(zhí)行defer語句
    */
    for {
        if <-closed {
            return
        }
    }
}

func main() {

    /**
    建立監(jiān)聽鏈接
    */
    ln, err := net.Listen("tcp", "127.0.0.1:6010")
    if err != nil {
        panic(err)
    }

    //創(chuàng)建一個管道

    //talkChan := map[f]
    talkChan := make(map[int]chan string)

    fmt.Printf("%p\n", talkChan)

    /*
       監(jiān)聽是否有客戶端過來的連接請求
    */
    for {
        fmt.Println("wait connect...")
        conn, err := ln.Accept()
        if err != nil {
            log.Fatal("get client connection error: ", err)
        }

        go handleConnection(conn, talkChan)
    }
}

客戶端測試代碼:client.go

復(fù)制代碼 代碼如下:

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)文章

  • Windows系統(tǒng)中搭建Go語言開發(fā)環(huán)境圖文詳解

    Windows系統(tǒng)中搭建Go語言開發(fā)環(huán)境圖文詳解

    GoLand?是?JetBrains?公司推出的商業(yè)?Go?語言集成開發(fā)環(huán)境(IDE),這篇文章主要介紹了Windows系統(tǒng)中搭建Go語言開發(fā)環(huán)境詳解,需要的朋友可以參考下
    2022-10-10
  • 淺析go中Ticker,Timer和Tick的用法與區(qū)別

    淺析go中Ticker,Timer和Tick的用法與區(qū)別

    在go面試的時候,面試官經(jīng)常會問time包的Ticker,Timer以及Tick的區(qū)別,一般在超時控制的時候用的比較多,今天就跟隨小編一起來詳細(xì)學(xué)一下這幾個的區(qū)別吧
    2023-10-10
  • 詳解Go語言微服務(wù)開發(fā)框架之Go chassis

    詳解Go語言微服務(wù)開發(fā)框架之Go chassis

    分布式系統(tǒng)中每個進(jìn)程的動態(tài)配置管理及運(yùn)行時熱加載就成為了一個亟待解決的問題。go chassis汲取了netflix的archaius框架經(jīng)驗,并做出來自己的創(chuàng)新特性。
    2021-05-05
  • go使用consul實(shí)現(xiàn)服務(wù)發(fā)現(xiàn)及配置共享實(shí)現(xiàn)詳解

    go使用consul實(shí)現(xiàn)服務(wù)發(fā)現(xiàn)及配置共享實(shí)現(xiàn)詳解

    這篇文章主要為大家介紹了go使用consul實(shí)現(xiàn)服務(wù)發(fā)現(xiàn)及配置共享實(shí)現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-05-05
  • Go?gRPC服務(wù)客戶端流式RPC教程

    Go?gRPC服務(wù)客戶端流式RPC教程

    這篇文章主要為大家介紹了Go?gRPC服務(wù)客戶端流式RPC教程示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-06-06
  • 詳解Go中的高效切片拼接和Go1.22提供的新方法

    詳解Go中的高效切片拼接和Go1.22提供的新方法

    在?Go?語言中,切片拼接是一項常見的操作,但如果處理不當(dāng),可能會導(dǎo)致性能問題或意外的副作用,本文將詳細(xì)介紹幾種高效的切片拼接方法,希望對大家有所幫助
    2024-01-01
  • 使用Go語言判斷二叉樹是否對稱的方法小結(jié)

    使用Go語言判斷二叉樹是否對稱的方法小結(jié)

    二叉樹Binary Tree一種特殊的樹,是結(jié)點(diǎn)的一個有限集合,且所有結(jié)點(diǎn)最多有2個子結(jié)點(diǎn),即度只能是0,1,2,判斷二叉樹是否對稱需比較左右子樹結(jié)構(gòu)與值,遞歸法直接對比子節(jié)點(diǎn),迭代法用隊列模擬遞歸,本文通過代碼示例講解的非常詳細(xì),需要的朋友可以參考下
    2025-07-07
  • go中值傳遞和指針傳遞的使用

    go中值傳遞和指針傳遞的使用

    在Go語言中,使用&和*可以分別取得變量的地址和值,解引用未初始化或為nil的指針會引發(fā)空指針異常,正確的做法是先進(jìn)行nil檢查,此外,nil在Go中用于多種類型的空值表示,值傳遞和指針傳遞各有適用場景,通常小型數(shù)據(jù)結(jié)構(gòu)優(yōu)先考慮值傳遞以減少解引用開銷
    2024-10-10
  • Go語言實(shí)現(xiàn)二分查找方法示例

    Go語言實(shí)現(xiàn)二分查找方法示例

    這篇文章主要為大家介紹了Go語言實(shí)現(xiàn)二分查找方法示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-12-12
  • 淺談go語言閉包的立即執(zhí)行和不立即執(zhí)行

    淺談go語言閉包的立即執(zhí)行和不立即執(zhí)行

    Go語言中的閉包是一種可以訪問其定義時所在作用域變量的特殊函數(shù),閉包可以分為立即執(zhí)行和不立即執(zhí)行兩種,本文就來介紹一下go語言閉包的立即執(zhí)行和不立即執(zhí)行,感興趣的可以了解一下
    2025-03-03

最新評論