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

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

 更新時(shí)間: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)聽(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去專門用來(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

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

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類型
        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 新開的線程去處理的。 |  即便chan阻塞,后面的也會(huì)執(zhí)行去讀 closed 這個(gè)chan
            }

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

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

        }
    }()

    /*
        從chan 里讀出給這個(gè)客戶端的數(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語(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

復(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語(yǔ)言開發(fā)環(huán)境圖文詳解

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

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

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

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

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

    分布式系統(tǒng)中每個(gè)進(jìn)程的動(dòng)態(tài)配置管理及運(yùn)行時(shí)熱加載就成為了一個(gè)亟待解決的問(wèn)題。go chassis汲取了netflix的archaius框架經(jīng)驗(yàn),并做出來(lái)自己的創(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?語(yǔ)言中,切片拼接是一項(xiàng)常見(jiàn)的操作,但如果處理不當(dāng),可能會(huì)導(dǎo)致性能問(wèn)題或意外的副作用,本文將詳細(xì)介紹幾種高效的切片拼接方法,希望對(duì)大家有所幫助
    2024-01-01
  • 使用Go語(yǔ)言判斷二叉樹是否對(duì)稱的方法小結(jié)

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

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

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

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

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

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

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

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

最新評(píng)論