今天讲讲TCP(详解)

本文涉及的产品
容器服务 Serverless 版 ACK Serverless,952元额度 多规格
容器服务 Serverless 版 ACK Serverless,317元额度 多规格
简介: 今天讲讲TCP(详解)

首先先说说什么是TCP?

TCP(Transmission Control Protocol,传输控制协议)是一种计算机通信协议,用于在互联网上可靠地传输数据。它是互联网协议(IP)的一部分,常用于应用层协议(如HTTP、FTP等)进行数据传输。

TCP为应用程序提供了一种面向连接的、可靠的数据传输服务。在使用TCP协议进行通信时,发送端和接收端先建立一条连接,然后通过这条连接进行数据传输。TCP协议会对传输的数据进行分段并对每个分段进行编号和校验,以确保数据在传输过程中不会丢失或被损坏。如果某个分段未能成功到达接收端,TCP协议会自动重新发送该分段,直到接收端正确地接收了所有数据。

另外,TCP协议还具有流量控制和拥塞控制的功能,可以在网络拥塞时自动减少发送速率以避免网络崩溃。这些功能使得TCP成为一种可靠的数据传输协议,广泛应用于互联网上各种需要可靠传输的应用程序中

一、TCP 协议基础

1.1、TCP 协议特点

       TCP 协议最主要的特点如下:

       面向连接:应用程序在使用 TCP 协议之前,必须先建立 TCP 连接。

       可靠性:TCP 提供可靠交付的服务。通过 TCP 连接传送的数据,无差错、不丢失、不重复,并且按序到达。如果数据包丢失或出现差错,则 TCP 负责重发数据。

       有序性:TCP 能够把发送的数据划分成一个个数据块,编号后发送,接收方根据编号将这些数据块组装成完整的数据。因此,在接收端可以确保数据块按照发送的顺序进行组装。

       流量控制:TCP 还提供了流量控制的功能,保证发送方的发送速度不会过快,导致接收方处理不及时,从而导致数据丢失。发送方会根据接收方返回的确认信息,调整自己的发送速度。

       拥塞控制:TCP 能够根据网络状况调整传输数据的速率,防止出现拥塞。如果网络出现拥塞,TCP 会通过降低发送方的数据传输速率和进行重传等措施来保证数据的可靠传输。

1.2、TCP 协议数据传输流程

       TCP 协议的数据传输流程包括三个步骤:

       1.连接建立:在数据传输之前,需要先建立连接。TCP 的连接建立采用“三次握手”协议,即由客户端发起连接请求,服务器返回确认信号,客户端再次返回确认信号,连接才算建立成功。

       2.数据传输:当 TCP 连接建立之后,数据就可以进行传输了。对于需要传输的数据,TCP 会将其分成一个个数据包进行传输,并且标记每一个数据包的序号,保证数据按照指定的顺序进行组装。同时,TCP 还会进行数据的可靠性检测,如果发现某个数据包没有被接收方正确接收,则会进行重传。

       3.连接终止:当数据传输完成时,TCP 需要进行连接的释放,以便释放资源。TCP 的连接释放采用“四次挥手”协议,即由客户端向服务器发送连接释放请求,服务器发送确认信号,服务器向客户端发送连接释放请求,客户端发送确认信号,连接才算全部释放。

       

二、TCP协议的工作原理

       TCP协议工作在OSI七层模型的第四层,即传输层。TCP协议通过三次握手建立连接,在连接建立后进行数据传输,并通过四次挥手关闭连接。

2.1、TCP的三次握手

       TCP连接的建立需要进行三次握手,握手的过程如下:

(1)客户端向服务器发送SYN报文,请求建立连接。

(2)服务器收到SYN请求报文并发回SYN+ACK报文,表示可以建立连接。

(3)客户端收到SYN+ACK报文后再发送ACK报文,表示连接建立成功。

       在三次握手过程中,SYN是同步序列号(Synchronize Sequence Number)的缩写,ACK是确认序列号(Acknowledgment)的缩写。

2.2、TCP的数据传输

       TCP协议通过标记每个数据包的序列号来保证数据的有序传输。TCP协议还提供了流量控制,确保数据在传输过程中不会被丢弃或超时。

       TCP协议还提供了错误检测和重传机制。如果数据包在传输过程中发生错误,接收方可以通过校验和来检测出这个错误,并要求发送方重新发送这个数据包。

三、TCP 协议的使用

3.1、建立 TCP 连接

       在 Java 中,可以使用 Socket 类来建立 TCP 连接。建立连接的流程如下:

       实例化 Socket 对象:可以使用带有主机名和端口号参数的构造器来新建 Socket 对象。

建立连接:使用 Socket 类中的 connect() 方法来建立连接。

下面是一个简单的TCP代码:

package com.example.tcpdemo;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.net.InetAddress;
import java.net.Socket;
public class TcpClient {
    public static void main(String[] args) {
        String serverHost = "localhost";
        int serverPort = 8080;
        Socket socket = null;
        BufferedReader reader = null;
        Writer writer = null;
        try {
            // 1.实例化 Socket 对象
            InetAddress inetAddress = InetAddress.getByName(serverHost);
            socket = new Socket(inetAddress, serverPort);
            // 2.建立连接
            reader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
            writer = new OutputStreamWriter(socket.getOutputStream());
            writer.write("Hello World!");
            writer.flush();
            // 3.处理服务器返回的数据
            System.out.println(reader.readLine());
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            try {
                if (socket != null) {
                    socket.close();
                }
                if (reader != null) {
                    reader.close();
                }
                if (writer != null) {
                    writer.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

 

3.2、发送数据

       在建立连接成功之后,可以使用 Socket 的 getOutputStream() 方法获取到输出流,通过输出流向服务器发送数据。

// 获取输出流
OutputStream outputStream = socket.getOutputStream();
// 发送数据
String data = "Hello World!";
outputStream.write(data.getBytes());
3.3、接收数据

在建立连接成功之后,可以使用 Socket 的 getInputStream() 方法获取到输入流,通过输入流接收从服务器返回的数据。

// 获取输入流
InputStream inputStream = socket.getInputStream();
// 接收数据
byte[] dataBuffer = new byte[1024];
int len = inputStream.read(dataBuffer);
String data = new String(dataBuffer, 0, len);

其中,dataBuffer 是缓冲区,len 表示读取的字节数,data 表示接收到的数据。

3.4、释放连接

TCP 连接释放需要经过“四次挥手”的步骤。在 Java 中,可以使用 Socket 类的 close() 方法来释放连接。

socket.close();

四、TCP 协议的问题和优化

4.1、TCP 协议的问题

       在网络延迟较大的情况下,TCP 需要等待控制信息的确认才能发送下一个数据包,导致效率较低。

       TCP 中的流控制和拥塞控制需要进行大量的计算,导致协议复杂性较高。

       当网络中出现丢包时,TCP 会尝试重传数据包,但是如果丢失的数据包较多,则重传的时间会变得非常长,影响了协议的效率。

4.2、TCP 协议的优化

使用快速重传机制:在接收方收到一个乱序的数据包时,可以立即向发送方发送一个 ACK 接收确认信息,要求其重新发送丢失的数据包。

使用加速算法:通过预测下一次数据包的到达时间,可以避免 TCP 因等待 ACK 而导致发送速度下降的问题,提高了协议的效率。

采用累积确认机制:TCP 可以将多个数据包合并成一组,一起进行确认,减少了发送的控制信息数量,提高了协议的效率。

五、总结

TCP 协议是互联网中的一种主要的传输协议,它具有面向连接、可靠性、有序性、流量控制和拥塞控制等特点。在使用时,需要先建立连接,进行数据传输,最后释放连接。同时,TCP 协议也存在一些问题,可以通过使用快速重传机制、加速算法和累积确认机制等方法来进行优化。


相关实践学习
通过Ingress进行灰度发布
本场景您将运行一个简单的应用,部署一个新的应用用于新的发布,并通过Ingress能力实现灰度发布。
容器应用与集群管理
欢迎来到《容器应用与集群管理》课程,本课程是“云原生容器Clouder认证“系列中的第二阶段。课程将向您介绍与容器集群相关的概念和技术,这些概念和技术可以帮助您了解阿里云容器服务ACK/ACK Serverless的使用。同时,本课程也会向您介绍可以采取的工具、方法和可操作步骤,以帮助您了解如何基于容器服务ACK Serverless构建和管理企业级应用。 学习完本课程后,您将能够: 掌握容器集群、容器编排的基本概念 掌握Kubernetes的基础概念及核心思想 掌握阿里云容器服务ACK/ACK Serverless概念及使用方法 基于容器服务ACK Serverless搭建和管理企业级网站应用
相关文章
|
2月前
|
网络协议 Java
谈谈TCP/IP网络编程
【9月更文挑战第1天】在当今数字化的世界中,网络通信是连接各种设备和系统的关键。TCP/IP协议作为互联网通信的基石,被广泛应用于各种网络场景。了解TCP/IP网络编程的概念,并掌握如何在Java中实现TCP/IP通讯,对于开发人员来说是非常重要的。
71 4
|
4月前
|
监控 网络协议 程序员
不再困惑!一文搞懂TCP与UDP的所有区别
**TCP与UDP是网络协议,TCP提供可靠连接(面向连接、顺序传输、错误检查),适合HTTP、FTP、SMTP等需要数据完整性的应用。UDP则是无连接、快速但不可靠,常用于DNS、RIP、SNMP等实时或效率优先的场景。**
219 0
|
5月前
|
监控 网络协议 安全
TCP和UDP面试题提问
TCP是一种面向连接、可靠的协议,提供确认和重传机制,确保数据完整性和可靠性,适合网页浏览、邮件收发等。UDP则是无连接、轻量级协议,不保证数据可靠性,但适合实时应用如语音视频通话和在线游戏,追求低延迟。
|
5月前
|
缓存 网络协议 算法
你从未见过如此详细的 TCP 八股文!
重传丢失的数据包; 如果再收到重复的 ACK,那么拥塞窗口值加 1; 如果收到新数据的 ACK 后,把拥塞窗口值设置为慢启动门限值,原因是该 ACK 确认了新的数据,说明丢失的数据包已收到,快速恢复过程结束,再次进入拥塞避免状态; 没有像超时重传一夜回到
99 3
|
缓存 网络协议 安全
|
网络协议 前端开发 算法
前端面试之TCP与UDP区别
前端面试之TCP与UDP区别
114 0
|
缓存 网络协议 网络性能优化
TCP连接是干什么的?底层原理是什么?
TCP连接是干什么的?底层原理是什么?
309 0
|
网络协议 测试技术
软件测试面试题:tcp 和 udp 的区别?
软件测试面试题:tcp 和 udp 的区别?
107 0
|
Kubernetes 网络协议 Dubbo
白话TCP/IP原理
白话TCP/IP原理
白话TCP/IP原理
|
网络协议 API 网络架构
40 张图带你搞懂 TCP 和 UDP(一)
这一篇文章是计算机网络连载文章的第四篇,历史文章请阅读 拿下计网协议后,我就是公园里最靓的仔 TCP/IP 基础知识总结 计算机网络基础知识总结 那么下面就开始我们本篇文章,文章组织脉络如下
40 张图带你搞懂 TCP 和 UDP(一)