婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > go語言實現(xiàn)聊天服務器的示例代碼

go語言實現(xiàn)聊天服務器的示例代碼

熱門標簽:南通如皋申請開通400電話 高德地圖標注口訣 學海導航地圖標注 江西轉化率高的羿智云外呼系統(tǒng) 中國地圖標注省會高清 西部云谷一期地圖標注 地圖標注的汽車標 廣州呼叫中心外呼系統(tǒng) 浙江高速公路地圖標注

看了兩天 go 語言,是時候練練手了。

go 的 routine(例程) 和 chan(通道) 簡直是神器,實現(xiàn)多線程(在 go 里準確的來說是 多例程)簡直不要太輕松。

于是動手碼了一個傻瓜版的黑框聊天器。

server 端:

監(jiān)聽 TCP 連接;支持自定義客戶端命令;支持消息分發(fā);理論上支持廣播;...

package main

import (
  "fmt"
  "net"
  "io"
  "strconv"
  "time"
  "strings"
)

const (
  NORMAL_MESSAGE = iota
  LIST_MESSAGE
)

var clientSenders = make(map[string] chan string)

func send (addr string, conn *net.Conn){
  senderChan := clientSenders[addr]
  for s := range senderChan{
    (*conn).Write([]byte(s))
  }
}

func sendUsersInfo(addr string){
  senderChan := clientSenders[addr]
  if nil != senderChan{
    ls := strconv.Itoa(LIST_MESSAGE)
    cs := strconv.Itoa(NORMAL_MESSAGE) + "已登錄客戶端列表:\n"
    i := 1
    for k := range clientSenders{
      a := ""
      if k == addr {
        a = "(我)"
      }
      cs = cs + strconv.Itoa(i) + ")" + k + a + "\n"
      ls += k + "\n"
      i ++
    }
    cs += "發(fā)送消息,可使用 1-這是給1號客戶端的消息\n(請使用英文以獲取最佳體驗)\n"

    senderChan - cs
    time.Sleep(time.Millisecond * 300)
    senderChan - ls

    // 發(fā)送格式化的列表

    fmt.Println("已發(fā)送“登錄用戶信息”", addr)
  } else{
    fmt.Println("客戶端接受通道不存在", addr)
  }
}

func serve (conn *net.Conn){
  connect := *conn

  addr := connect.RemoteAddr().String()

  fmt.Println(addr, "接入服務")

  senderChan := make(chan string, 3)
  clientSenders[addr] = senderChan

  // 啟動發(fā)送
  go send(addr, conn)

  // 發(fā)送當前用戶信息
  go sendUsersInfo(addr)

  buff := make([]byte, 10240)
  for {
    n, err := connect.Read(buff)
    if err != nil {
      if err == io.EOF {
        fmt.Println("客戶端斷開鏈接,", addr)
        delete(clientSenders, addr)
        return
      } else{
        fmt.Println(err)
      }
    }

    msg := string(buff[:n])

    // 刷新客戶端列表
    if msg == "ls\n" {
      go sendUsersInfo(addr)
      continue
    }

    // 提取數(shù)據(jù)
    msgs := strings.Split(msg, "-")
    if len(msg)  2{
      senderChan - string("數(shù)據(jù)格式不正確,請聯(lián)系開發(fā)者")
      continue
    }

    aimAddr := msgs[0]
    aimSender := clientSenders[aimAddr]
    if aimSender == nil {
      senderChan - string("客戶端已下線,使用 ls 命令獲取最新的客戶端列表")
      continue
    }

    aimSender - strconv.Itoa(NORMAL_MESSAGE) + "[from:" + addr + "]:" + strings.Join(msgs[1:], "-")
  }
}

func main(){
  addr := ":8080"
  listener, err := net.Listen("tcp", addr)
  if err != nil{
    fmt.Println(err)
    return
  }

  // 啟動消息調度器

  defer listener.Close()

  // 啟動連接監(jiān)聽
  for {
    conn, err := listener.Accept()
    if err != nil {
      fmt.Println(err)
      continue
    }

    go serve(conn)
  }
}

客戶端:

支持斷線重連;支持給特定其他客戶端發(fā)信息

package main

import (
  "net"
  "fmt"
  "io"
  "os"
  "bufio"
  "sync"
  "time"
  "strings"
  "strconv"
)


var conn *net.Conn
var addrs []string

const (
  NORMAL_MESSAGE = iota
  LIST_MESSAGE
)

func read(conn2 *net.Conn){
  defer func() {
    fmt.Println("嘗試重連")
    go connectServer()
  }()

  connect := *conn2
  buff := make([]byte, 20140)
  for {
    n, err := connect.Read(buff)
    if err != nil {
      if err == io.EOF{
        fmt.Println("結束")
        (*conn2).Close()
        conn = nil
        return
      } else{
        fmt.Println(err)
      }
    }

    msg := string(buff[:n])
    t, err := strconv.Atoi(string(msg[0]))
    msg = msg[1:]

    switch t {
    case NORMAL_MESSAGE:
      fmt.Print(msg)
      break
    case LIST_MESSAGE:
      // 解析客戶端列表數(shù)據(jù)
      addrs = strings.Split(msg, "\n")
      fmt.Println("已接收客戶端列表。\n")
      break
    default:
      fmt.Print(msg)
      break
    }
  }
}

func connectServer(){
  addr := "192.168.99.236:8080"
  fmt.Println("等待服務器開啟中")
  conn2, err := net.Dial("tcp", addr)
  if err != nil {
    fmt.Print(err)
    fmt.Println("連接失敗,10s后嘗試")
    time.Sleep(10 * time.Second)
    go connectServer()
    return
  }

  fmt.Println("已連接")

  conn = conn2
  go read(conn2)
}

func send (){
  inputReader := bufio.NewReader(os.Stdout)
  for {
    input, err := inputReader.ReadString('\n')
    if err != nil {
      if err == io.EOF{
        return
      } else{
        fmt.Println(err)
      }
    }

    if input == "ls\n" {
      (*conn).Write([]byte(input))
      continue
    }

    msgs := strings.Split(input, "-")
    if len(msgs)  2 {
      fmt.Println("發(fā)送的姿勢不正確,應該像這樣 1-給1號發(fā)送消息\n")
      continue
    }

    index, err := strconv.Atoi(msgs[0])
    if err != nil {
      fmt.Println("發(fā)送的姿勢不正確,應該像這樣 1-給1號發(fā)送消息\n")
      continue
    }

    if len(addrs) = index {
      fmt.Println("不存在第" + strconv.Itoa(index) + "個客戶端\n")
      continue
    }

    addr := addrs[index-1]

    input = addr + "-" + strings.Join(msgs[1:], "-")

    if nil != conn {
      (*conn).Write([]byte(input))
    }
  }
}

func main (){
  var wg sync.WaitGroup
  wg.Add(2)
  go connectServer()
  go send()
  wg.Wait()

  defer func() {
    if nil != conn {
      (*conn).Close()
    }
  }()
}

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • 淺談Golang中創(chuàng)建一個簡單的服務器的方法
  • Go語言的變量、函數(shù)、Socks5代理服務器示例詳解
  • Go語言使用HTTP包創(chuàng)建WEB服務器的方法
  • golang實現(xiàn)http服務器處理靜態(tài)文件示例
  • 剖析Go編寫的Socket服務器模塊解耦及基礎模塊的設計
  • 服務器端Go程序對長短鏈接的處理及運行參數(shù)的保存
  • Go語言基于Socket編寫服務器端與客戶端通信的實例

標簽:東營 曲靖 吐魯番 許昌 貴州 常州 德宏 保定

巨人網絡通訊聲明:本文標題《go語言實現(xiàn)聊天服務器的示例代碼》,本文關鍵詞  語言,實現(xiàn),聊天,服務器,;如發(fā)現(xiàn)本文內容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內容系統(tǒng)采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《go語言實現(xiàn)聊天服務器的示例代碼》相關的同類信息!
  • 本頁收集關于go語言實現(xiàn)聊天服務器的示例代碼的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 崇明县| 泸西县| 克什克腾旗| 武宁县| 余江县| 秦皇岛市| 左云县| 常州市| 裕民县| 铜陵市| 潮州市| 滕州市| 东光县| 衢州市| 大渡口区| 吴江市| 葫芦岛市| 广汉市| 洞口县| 太原市| 且末县| 涪陵区| 崇明县| 江永县| 清丰县| 高安市| 东乌| 怀远县| 安平县| 凯里市| 息烽县| 天柱县| 大洼县| 石屏县| 沿河| 奉贤区| 泰兴市| 甘孜县| 四会市| 巴林右旗| 石嘴山市|