网络编程

网络编程基本介绍

  1. Golang 的主要设置目标之一就是面向大规模后端服务程序,网络通信这块是程序必不可少也是至关重要的一部分。
  2. 网络编程有两种:
    • TCP Socket 编程,是网络编程的主流。之所以叫 TCP Socket 编程,就是因为底层是基于Tcp/Ip协议的,比如:QQ 聊天。
    • b/s 结构的 http 编程,我们使用浏览器去访问服务器时,使用的就是 http 协议,而 http 底层仍旧使用 Tcp Socket实现的。比如:京东商城。
  3. 计算机间要相互通讯,必须要求网线,网卡,或者是无线网卡。
  4. TCP/IP(Transmission Control Protocol/Internet Protocol)的简写,中文译名为传输控制协议/因特网互联协议,又叫网络通讯协议,这个协议是 Internet 最基本的协议、Internet 国际互联网络的基础,简单地说,就是由网络层的 IP 协议和传输层的 TCP 协议组成的。

    OSI 模型

  5. OSI 模型包括应用层(application)、表示层(presention)、会话层(session)、传输层(transport)、网络层(ip)、数据链路层(link)、
    物理层(physical)。

    IP 地址

  6. 概述:每个 internet 上的主机和路由器都有一个 ip 地址,它包括网络号和主机号,地址有 ipv4(32位)或者 ipv6(128位)。可以通过 ipconfig/ifconfig(Mac) 来查看。

    端口

  7. 我们这里所指的端口不是物理意义上的端口,而是特指 TCP/IP 协议中的端口,是逻辑意义上的端口。

  8. 如果把 IP 地址比作一间房子,端口就是出入这件房子的门。真正的房子只有几个门,但是一个 IP 地址的端口 可以有 65536(即:256 x 256)个之多!
  9. 端口是通过端口号来标记,端口号只有证书,范围是从0到65535(256 x 256-1)。
  10. 只要是做服务程序,都必须监听一个端口。
  11. 该端口就是其他程序和该服务通讯的通道。
  12. 一台电脑有65535个端口 1-65535。
  13. 一旦一个端口被某个程序监听(占用),那么其他的程序就不能在该端口监听。

    端口分类

  14. 0 是保留端口

  15. 1-1024 是固定端口,又叫有名端口,即被某些程序固定使用,一般程序员不使用。
    • 22:SSH 远程登录协议
    • 23:telnet 使用
    • 21:ftp 使用
    • 25:smtp 服务使用
    • 80:iis 使用
    • 7: echo 服务
  16. 1025-65535 是动态端口,这些端口程序员可以使用。

    端口使用注意事项

  17. 在计算机(尤其是做服务器)要尽可能的少开端口

  18. 一个端口只能被一个程序监听
  19. 如果使用 netstat -an 可以查看本机有哪些端口在监听
  20. 可以使用 netstat -anb 来查看监听端口的 pid,在结合任务管理器关闭不安全的端口。

    Tcp Socket 编程的快速入门

    服务端的处理流程

  21. 监听端口

  22. 接收客户端的 tcp 链接,建立客户端和服务器端的连接
  23. 创建 goroutine ,处理该链接的请求(通常客户端会通过链接发送请求包)

    客户端的处理流程

  24. 建立与服务端的链接

  25. 发送请求数据,接收服务器端返回的结果数据
  26. 关闭链接

    快速入门应用实例

  27. 服务器端功能:

    • 编写一个服务器端程序,在8888端口监听
    • 可以和多个客户端创建链接
    • 链接成功后,客户端可以发送数据,服务端接收数据,并显示在终端上
    • 先使用 Telnet 来测试,然后编写客户端程序来测试
  28. 客户端功能:
    • 编写一个客户端端程序,能链接到服务器端的 8888 端口
    • 客户端可以发送单行数据,然后就退出
    • 能通过终端输入数据(输入一行发送一行),并发送给服务端
    • 在终端输入 exit,表示退出程序
  29. 代码实现 ```go //server.go package main

import ( “fmt” “io” “net” //做网络 socket 开发,net 包含我们需要所有的方法核函数。 )

func process(conn net.Conn) { //这里我们循环的接收客户端发送的数据 defer conn.Close() // 延时关闭 conn

  1. for {
  2. //创建一个新的切片
  3. buf := make([]byte, 1024)
  4. // conn.Read(buf)
  5. //1. 等待客户端通过 conn 发送信息
  6. //2. 如果客户端没有 write,那么协程阻塞在这里
  7. // fmt.Printf("服务器在等待客户端 %s 发送信息 \n", conn.RemoteAddr().String())
  8. n, err := conn.Read(buf) //从 conn 读取
  9. if err == io.EOF {
  10. fmt.Println("客户端退出")
  11. return
  12. } else {
  13. // 显示客户端发送的内容到服务器的终端
  14. fmt.Print(string(buf[:n]))
  15. }
  16. }

}

func main() {

  1. fmt.Println("服务器开始监听...")
  2. // listen, err := net.Listen("tcp", "127.0.0.1:8888") // 这种写法只支持 IPv4
  3. //1. tcp 表示使用网络协议是 tcp
  4. //2. 0.0.0.0:8888 表示在本地监听 8888 端口
  5. listen, err := net.Listen("tcp", "0.0.0.0:8888")
  6. if err != nil {
  7. fmt.Println("listen err =", err)
  8. return
  9. }
  10. defer listen.Close() // 延时关闭 listen
  11. //循环等待客户端来链接
  12. for {
  13. //等待客户端来链接
  14. fmt.Println("等待客户端来链接...")
  15. conn, err := listen.Accept()
  16. if err != nil {
  17. fmt.Println("Accept() err =", err)
  18. } else {
  19. fmt.Printf("Accept() conn = %v 客户端 IP = %v \n", conn, conn.RemoteAddr().String())
  20. }
  21. // 这起协程,为客户端服务
  22. go process(conn)
  23. }
  24. // fmt.Printf("listen = %v \n", listen)

}

  1. ```go
  2. //client.go
  3. package main
  4. import (
  5. "bufio"
  6. "fmt"
  7. "net"
  8. "os"
  9. "strings"
  10. )
  11. func main() {
  12. conn, err := net.Dial("tcp", "127.0.0.1:8888")
  13. if err != nil {
  14. fmt.Println("Client dial err =", err)
  15. return
  16. }
  17. //fmt.Println("conn =", conn)
  18. // 功能一:客户端可以发送单行数据,然后就退出
  19. reader := bufio.NewReader(os.Stdin) // os.Stdin 代表标准输入(终端输入)
  20. for {
  21. // 从终端读取一行用户输入,并准备发送给服务器
  22. line, err := reader.ReadString('\n')
  23. if err != nil {
  24. fmt.Println("readString err =", err)
  25. } else {
  26. line = strings.Trim(line, " \r\n")
  27. if line == "exit" {
  28. fmt.Println("客户端退出...")
  29. break
  30. }
  31. //再将 line 发送给服务器
  32. n, err := conn.Write([]byte(line + "\n"))
  33. if err != nil {
  34. fmt.Println("conn.Write err =", err)
  35. } else {
  36. fmt.Printf("客户端发送了 %d 字节的数据 \n", n)
  37. }
  38. }
  39. }
  40. }

课程来源