12.2.1 什么是Socket

Socket起源于Unix,而Unix基本哲学之一就是“一切皆文件”,都可以用“打开open –> 读写write/read –> 关闭close”模式来操作。Socket就是该模式的一个实现,网络的Socket数据传输是一种特殊的I/O,Socket也是一种文件描述符。Socket也具有一个类似于打开文件的函数调用:Socket(),该函数返回一个整型的Socket描述符,随后的连接建立、数据传输等操作都是通过该Socket实现的。

常用的Socket类型有两种:流式Socket(SOCK_STREAM)和数据报式Socket(SOCK_DGRAM)。流式是一种面向连接的Socket,针对于面向连接的TCP服务应用;数据报式Socket是一种无连接的Socket,对应于无连接的UDP服务应用。

12.2.2 TCP的C/S架构

12.2.3 示例程序

12.2.3.1 服务器代码

  1. package main
  2. import (
  3. "fmt"
  4. "log"
  5. "net"
  6. "strings"
  7. )
  8. func dealConn(conn net.Conn) {
  9. defer conn.Close() //此函数结束时,关闭连接套接字
  10. //conn.RemoteAddr().String():连接客服端的网络地址
  11. ipAddr := conn.RemoteAddr().String()
  12. fmt.Println(ipAddr, "连接成功")
  13. buf := make([]byte, 1024) //缓冲区,用于接收客户端发送的数据
  14. for {
  15. //阻塞等待用户发送的数据
  16. n, err := conn.Read(buf) //n代码接收数据的长度
  17. if err != nil {
  18. fmt.Println(err)
  19. return
  20. }
  21. //切片截取,只截取有效数据
  22. result := buf[:n]
  23. fmt.Printf("接收到数据来自[%s]==>[%d]:%s\n", ipAddr, n, string(result))
  24. if "exit" == string(result) { //如果对方发送"exit",退出此链接
  25. fmt.Println(ipAddr, "退出连接")
  26. return
  27. }
  28. //把接收到的数据转换为大写,再给客户端发送
  29. conn.Write([]byte(strings.ToUpper(string(result))))
  30. }
  31. }
  32. func main() {
  33. //创建、监听socket
  34. listenner, err := net.Listen("tcp", "127.0.0.1:8000")
  35. if err != nil {
  36. log.Fatal(err) //log.Fatal()会产生panic
  37. }
  38. defer listenner.Close()
  39. for {
  40. conn, err := listenner.Accept() //阻塞等待客户端连接
  41. if err != nil {
  42. log.Println(err)
  43. continue
  44. }
  45. go dealConn(conn)
  46. }
  47. }

12.2.3.2 客户端代码

  1. package main
  2. import (
  3. "fmt"
  4. "log"
  5. "net"
  6. )
  7. func main() {
  8. //客户端主动连接服务器
  9. conn, err := net.Dial("tcp", "127.0.0.1:8000")
  10. if err != nil {
  11. log.Fatal(err) //log.Fatal()会产生panic
  12. return
  13. }
  14. defer conn.Close() //关闭
  15. buf := make([]byte, 1024) //缓冲区
  16. for {
  17. fmt.Printf("请输入发送的内容:")
  18. fmt.Scan(&buf)
  19. fmt.Printf("发送的内容:%s\n", string(buf))
  20. //发送数据
  21. conn.Write(buf)
  22. //阻塞等待服务器回复的数据
  23. n, err := conn.Read(buf) //n代码接收数据的长度
  24. if err != nil {
  25. fmt.Println(err)
  26. return
  27. }
  28. //切片截取,只截取有效数据
  29. result := buf[:n]
  30. fmt.Printf("接收到数据[%d]:%s\n", n, string(result))
  31. }
  32. }
作者:admin  创建时间:2018-06-18 07:16
 更新时间:2018-06-18 07:18
上一篇:
下一篇: