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

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

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

實現(xiàn)邏輯

1、Golang 版本  1.3

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

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

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

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

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

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

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

實現(xiàn)代碼

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

復制代碼 代碼如下:

package main

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

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

    /*
        定義當前用戶的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")

    //當前所有的連接
    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ù),寫到相應的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

復制代碼 代碼如下:

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

  • Golang創(chuàng)建構(gòu)造函數(shù)的方法超詳細講解

    Golang創(chuàng)建構(gòu)造函數(shù)的方法超詳細講解

    構(gòu)造器一般面向?qū)ο笳Z言的典型特性,用于初始化變量。Go語言沒有任何具體構(gòu)造器,但我們能使用該特性去初始化變量。本文介紹不同類型構(gòu)造器的差異及其應用場景
    2023-01-01
  • go中for?range的坑以及解決方案

    go中for?range的坑以及解決方案

    相信小伙伴都遇到過以下的循環(huán)變量的問題,本文主要介紹了go中for?range的坑以及解決方案,具有一定的參考價值,感興趣的可以了解一下
    2024-01-01
  • golang 實現(xiàn)每隔幾分鐘執(zhí)行一個函數(shù)

    golang 實現(xiàn)每隔幾分鐘執(zhí)行一個函數(shù)

    這篇文章主要介紹了golang 實現(xiàn)每隔幾分鐘執(zhí)行一個函數(shù),具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • go語言題解LeetCode989數(shù)組形式的整數(shù)加法

    go語言題解LeetCode989數(shù)組形式的整數(shù)加法

    這篇文章主要為大家介紹了go語言題解LeetCode989數(shù)組形式的整數(shù)加法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-12-12
  • Go語言使用組合的方式實現(xiàn)多繼承的方法

    Go語言使用組合的方式實現(xiàn)多繼承的方法

    這篇文章主要介紹了Go語言使用組合的方式實現(xiàn)多繼承的方法,實例分析了多繼承的原理與使用組合方式來實現(xiàn)多繼承的技巧,需要的朋友可以參考下
    2015-02-02
  • 一文幫你搞懂Go面試中常問的channel問題

    一文幫你搞懂Go面試中常問的channel問題

    channel是Golang面試時經(jīng)常會問到的問題,所以這篇文章為大家整理了channel??嫉囊恍﹩栴}以及回答,感興趣的小伙伴可以跟隨小編一起學習一下
    2023-06-06
  • Go語言中的sync包同步原語最新詳解

    Go語言中的sync包同步原語最新詳解

    Go語言在sync包中提供了一套多才多藝的同步機制,以及用于管理對共享資源的并發(fā)訪問的原子操作,了解這些工具并為您的并發(fā)需求選擇合適的工具是編寫高效可靠的并發(fā)Go程序的關(guān)鍵,這篇文章主要介紹了Go語言中的`sync`包同步原語,需要的朋友可以參考下
    2023-12-12
  • 基于Go?goroutine實現(xiàn)一個簡單的聊天服務(wù)

    基于Go?goroutine實現(xiàn)一個簡單的聊天服務(wù)

    對于聊天服務(wù),想必大家都不會陌生,因為在我們的生活中經(jīng)常會用到,本文我們用?Go?并發(fā)來實現(xiàn)一個聊天服務(wù)器,這個程序可以讓一些用戶通過服務(wù)器向其它所有用戶廣播文本消息,文中通過代碼示例介紹的非常詳細,需要的朋友可以參考下
    2023-06-06
  • Golang中crypto/rand庫的使用技巧與最佳實踐

    Golang中crypto/rand庫的使用技巧與最佳實踐

    在Golang的眾多隨機數(shù)生成庫中,crypto/rand?是一個專為加密安全設(shè)計的庫,本文主要介紹了Golang中crypto/rand庫的使用技巧與最佳實踐,感興趣的可以了解一下
    2024-02-02
  • 安裝GoLang環(huán)境和開發(fā)工具的圖文教程

    安裝GoLang環(huán)境和開發(fā)工具的圖文教程

    Go是一門由Google開發(fā)的編程語言,GoLand的安裝非常簡單,本文主要介紹了安裝GoLang環(huán)境和開發(fā)工具的圖文教程,具有一定的參考價值,感興趣的可以了解一下
    2023-09-09

最新評論