利用golang实现聊天通信

aqsmoke · · 1369 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

实现逻辑

1、Golang 版本  1.3

2、实现原理:

  1、主进程建立TCP监听服务,并且初始化一个变量 talkChan := make(map[int]chan string)

  2、当主进程ACCEPT连接请求后,利用go 启动一个协程A去维持和客户端的连接,把taokChan带入到协程里

  3、和客户端建立连接的协程A,发送消息给客户端,使其发送自己的用户信息。

  4、协程A在收到客户端发送的用户信息后,建立一个此用户对应的管道 talkChan[uid] = make(chan string)

  5、协程A再启动一个协程A1去专门用来读取客户端发送的消息,并且用来判断是发送给谁的消息,然后把消息放到对应的chan里。 

  6、协程A再启动一个协程A2用来读取此用户对应的管道,如果里面有信息,则取出来发送到客户端。

实现代码

 服务端测试代码:server.go

  1 package main
  2 
  3 import (
  4     "fmt"
  5     "log"
  6     "net"
  7     "strconv"
  8 )
  9 
 10 func handleConnection(conn net.Conn, talkChan map[int]chan string) {
 11     //fmt.Printf("%p\n", talkChan)  //用以检查是否是传过来的指针
 12 
 13     /*
 14         定义当前用户的uid
 15     */
 16     var curUid int
 17 
 18     var err error
 19 
 20     /*
 21         定义关闭通道
 22     */
 23     var closed = make(chan bool)
 24 
 25     defer func() {
 26         fmt.Println("defer do : conn closed")
 27         conn.Close()
 28         fmt.Printf("delete userid [%v] from talkChan", curUid)
 29         delete(talkChan, curUid)
 30     }()
 31 
 32     /**
 33      * 提示用户设置自己的uid, 如果没设置,则不朝下执行
 34      */
 35     for {
 36         //提示客户端设置用户id
 37         _, err = conn.Write([]byte("请设置用户uid"))
 38         if err != nil {
 39             return
 40         }
 41         data := make([]byte, 1024)
 42         c, err := conn.Read(data)
 43         if err != nil {
 44             //closed <- true  //这样会阻塞 | 后面取closed的for循环,没有执行到。
 45             return
 46         }
 47         sUid := string(data[0:c])
 48 
 49         //转成int类型
 50         uid, _ := strconv.Atoi(sUid)
 51         if uid < 1 {
 52             continue
 53         }
 54         curUid = uid
 55         talkChan[uid] = make(chan string)
 56         //fmt.Println(conn, "have set uid ", uid, "can talk")
 57 
 58         _, err = conn.Write([]byte("have set uid "+sUid+" can talk"))
 59         if err != nil {
 60             return
 61         }
 62         break
 63     }
 64 
 65     fmt.Println("err 3")
 66 
 67     //当前所有的连接
 68     fmt.Println(talkChan)
 69 
 70     //读取客户端传过来的数据
 71     go func() {
 72         for {
 73             //不停的读客户端传过来的数据
 74             data := make([]byte, 1024)
 75             c, err := conn.Read(data)
 76             if err != nil {
 77                 fmt.Println("have no client write", err)
 78                 closed <- true //这里可以使用 | 因为是用用的go 新开的线程去处理的。 |  即便chan阻塞,后面的也会执行去读 closed 这个chan
 79             }
 80 
 81             clientString := string(data[0:c])
 82 
 83             //将客户端过来的数据,写到相应的chan里
 84             if curUid == 3 {
 85                 talkChan[4] <- clientString
 86             } else {
 87                 talkChan[3] <- clientString
 88             }
 89 
 90         }
 91     }()
 92 
 93     /*
 94         从chan 里读出给这个客户端的数据 然后写到该客户端里
 95     */
 96     go func() {
 97         for {
 98             talkString := <-talkChan[curUid]
 99             _, err = conn.Write([]byte(talkString))
100             if err != nil {
101                 closed <- true
102             }
103         }
104     }()
105 
106     /*
107        检查是否已经关闭连接 如果关闭则推出该线程  去执行defer语句
108     */
109     for {
110         if <-closed {
111             return
112         }
113     }
114 }
115 
116 func main() {
117 
118     /**
119     建立监听链接
120     */
121     ln, err := net.Listen("tcp", "127.0.0.1:6010")
122     if err != nil {
123         panic(err)
124     }
125 
126     //创建一个管道
127 
128     //talkChan := map[f]
129     talkChan := make(map[int]chan string)
130 
131     fmt.Printf("%p\n", talkChan)
132 
133     /*
134        监听是否有客户端过来的连接请求
135     */
136     for {
137         fmt.Println("wait connect...")
138         conn, err := ln.Accept()
139         if err != nil {
140             log.Fatal("get client connection error: ", err)
141         }
142 
143         go handleConnection(conn, talkChan)
144     }
145 }

客户端测试代码:client.go

 1 package main
 2 
 3 import (
 4     "fmt"
 5     "math/rand"
 6     "net"
 7 )
 8 
 9 func main() {
10     conn, err := net.Dial("tcp", "127.0.0.1:6010")
11     if err != nil {
12         panic(err)
13     }
14 
15     fmt.Fprintf(conn, "hello server\n")
16 
17     defer conn.Close()
18     go writeFromServer(conn)
19 
20     for {
21         var talkContent string
22         fmt.Scanln(&talkContent)
23 
24         if len(talkContent) > 0 {
25             _, err = conn.Write([]byte(talkContent))
26             if err != nil {
27                 fmt.Println("write to server error")
28                 return
29             }
30         }
31     }
32 }
33 
34 func connect() {
35     conn, err := net.Dial("tcp", "127.0.0.1:6010")
36     if err != nil {
37         panic(err)
38     }
39 
40     fmt.Fprintf(conn, "hello server\n")
41 
42     defer conn.Close()
43     go writeFromServer(conn)
44 
45     for {
46         var talkContent string
47         fmt.Scanln(&talkContent)
48 
49         if len(talkContent) > 0 {
50             _, err = conn.Write([]byte(talkContent))
51             if err != nil {
52                 fmt.Println("write to server error")
53                 return
54             }
55         }
56     }
57 }
58 
59 func writeFromServer(conn net.Conn) {
60     defer conn.Close()
61     for {
62         data := make([]byte, 1024)
63         c, err := conn.Read(data)
64         if err != nil {
65             fmt.Println("rand", rand.Intn(10), "have no server write", err)
66             return
67         }
68         fmt.Println(string(data[0:c]) + "\n ")
69     }
70 }

 

本文来自:博客园

感谢作者:aqsmoke

查看原文:利用golang实现聊天通信

1369 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传