[JavaEE -- 传输层中UDP和TCP的api实现原理]

传输层中UDP和TCP的api实现原理

  • 1. UDP和TCP协议特点
    • 1.1 TCP
    • 1.2 UDP
  • 2. UDP协议中socket api的使用
    • 2.1 服务器:
    • 2.2 客户端
    • 2.3 整个流程
  • 3. TCP协议中的api使用
    • 3.1 TCP服务器
    • 3.2 TCP客户端
    • 3.3 整个流程

1. UDP和TCP协议特点

1.1 TCP

  • 有连接:抽象,虚拟的连接(如打电话)
  • 可靠传输:尽可能的完成数据传输,虽然无法确保数据到达对方,至少可以知道,当前数据是不是收到了
  • 面向字节流:此处的字节流和文件中的字节流完全一致
  • 全双工:一个通信可以双向通信。(而半双工只能单向通信)

1.2 UDP

  • 无连接:无论是否同意都会发送
  • 不可靠传输:不确定数据是否能收到
  • 面向数据报:每次传输的基本单位是一个数据报(由一系列的字节构成的)特定结构
  • 全双工

2. UDP协议中socket api的使用

操作系统中有一类文件,就叫做socket文件,抽象表示了 网卡 这样的硬件设备。进行网络通信最核心的硬件设备网卡,通过网卡发送数据,就是写socket文件,通过网卡接收数据,就是读socket文件
核心的类有两个:

  1. DatagramSocket
    DatagramSocket(),创建一个UDP数据报套接字的Socket,绑定到本机任意一个随机接口(一般用于客户端)
    DatagramSocket(port),创建一个UDP数据报套接字的Socket,绑定本机指定的端口(一般用于服务器)
    void receive(DatagramPacket p) :从此套接字接收数据报(如果没有接收数据报,该方法会阻塞等待)
    void send(DatagramPacket p) :从此套接字发送数据报(不会阻塞等待,直接发送)

  2. DatagramPacket
    UDP面向数据报,每次发送数据报的基本单位,就是UDP数据报。表示了一个UDP数据报。

2.1 服务器:

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.Socket;
import java.net.SocketException;
import java.nio.charset.StandardCharsets;


public class UdpEchoServer {
	//创建DatagramSocket对象
    private DatagramSocket socket = null;
    public UdpEchoServer(int port) throws SocketException { // 网络编程中常见的异常,通常表示socket创建失败,比如端口号已经被其他进程占用了,就会失败
        socket = new DatagramSocket(port);
    }

    // 服务器启动逻辑
    public void start() throws IOException {
        System.out.println("服务器启动!");
        // 对于服务器来说,需要不停的收到请求,返回响应
        while (true) {
            // 每次循环,就是处理一个请求-响应过程
            // 1. 读取请求并解析
            DatagramPacket requestPacket = new DatagramPacket(new byte[4096],4096);// 通过这个字节数组保存收到的消息正文(应用层数据包)也就是UDP数据报载荷部分
            // receive接收从网卡读取到一个UDP数据报,放入requestPacket对象中
            socket.receive(requestPacket);

            // 读到的字节数组,转成String方便后续逻辑处理
            String request = new String(requestPacket.getData(),0,requestPacket.getLength());
            // 2. 根据请求计算响应(对于 回显服务器来说,这一步啥也不做)
            String response = process(request);
            // 3. 把响应返回到客户端
            //      构建一个DatagramPacket 作为响应对象
            // requestPacket.getSocketAddress()  从requestPacket客户端来的数据报,获得INetAddress对象这个对象就包含了ip和端口号(和服务器通信对端)
            DatagramPacket responsePacket = new DatagramPacket(response.getBytes(), response.getBytes().length,
                                        requestPacket.getSocketAddress());
            socket.send(responsePacket);

            //打印日志
            System.out.printf("[%s:%d] req: %s,resp: %s\n",requestPacket.getAddress().toString(),
                    requestPacket.getPort(),request,response);
        }
    }

    public String process(String request) {
        return request;
    }

    public static void main(String[] args) throws IOException {
        UdpEchoServer udpEchoServer = new UdpEchoServer(9090);
        udpEchoServer.start();
    }
}

2.2 客户端


import java.io.IOException;
import java.net.*;
import java.util.Scanner;

public class UdpEchoClient {
    private DatagramSocket socket = null;
    private String serverIp;
    private int serverPort;


    // 此处的IP使用的字符串,点分十进制,”192.168.2.100“ 不能直接使用serverIp,要InetAddress.getByName(serverIp)
    public UdpEchoClient(String serverIP,int serverPort) throws SocketException {
        this.serverIp = serverIP;
        this.serverPort = serverPort;
        // 客户端一般不要手动指定端口号,系统会自动分配一个空闲的端口号
        socket = new DatagramSocket();
    }
    public void start() throws IOException {
        System.out.println("启动客户端");
        Scanner scanner = new Scanner(System.in);
        while (true) {
            System.out.print("-> ");
            // 1. 从控制台读取要发送的请求数据
            if (!scanner.hasNext()) {
                break;
            }
            String request = scanner.next();

            // 2. 构造请求并发送  (把string里面的内容作为请求)
            DatagramPacket requestPacket = new DatagramPacket(request.getBytes(),0,request.getBytes().length
            , InetAddress.getByName(serverIp),serverPort);
            socket.send(requestPacket);

            // 3, 获取服务器响应
            DatagramPacket responsePacket = new DatagramPacket(new byte[4096],4096);
            socket.receive(responsePacket);
            // 4. 把响应显示到控制台上
            String response = new String(responsePacket.getData(),0,responsePacket.getLength());
            System.out.println(response);
        }

    }
    public static void main(String[] args) throws IOException {
        UdpEchoClient client = new UdpEchoClient("127.0.0.1",9090);
        client.start();
    }
}

2.3 整个流程

  1. 服务器启动,启动之后立即进入while循环,执行到receive,进入阻塞等待客户端的请求
  2. 客户端启动之后,阻塞在hasNext等待用户在控制台输入
  3. 用户在客户端输入之后,客户端就拿到字符串构造出一个(requestPacket)请求,send发送请求,然后创建responsePacket来执行receive接收操作并且等待响应返回。
  4. 服务器收到请求之后,就会从receive的阻塞中返回,之后就会根据读到的DatagramPacket对象,构造String request,通过process方法构造一个String response,再根据response构建一个DatagramPacket表示响应对象,再通过send来进行发送给客户端。(这个过程中,客户端在阻塞等待)。
  5. 客户端从receive中返回执行,就能得到服务器返回的响应,并且打印到控制台上,与此同时,服务器进入下一次循环,也要进入到第二次的receive阻塞。等待下一个请求。

3. TCP协议中的api使用

  • ServerSocket:这个socket类对应到网卡,但是这个类只能给服务器进行使用
  • Socket:对应到网卡,既可以给服务器使用,也可以给客户端使用。

TCP是面向字节流的,传输的基本单位是字节,是有连接的,通过accept来完成连接,accept也是可能会产生阻塞的操作,如果当前客户端还没有连接过来,此时accept就会阻塞。

3.1 TCP服务器

//  ServerSocket 这是Socket类对应到网卡,但是这个类只能给服务器使用
import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;

public class TcpEchoServer {
    private ServerSocket serverSocket = null;
    public TcpEchoServer(int port) throws IOException {
        serverSocket = new ServerSocket(port);
    }

    public void start() throws IOException {
        System.out.println("服务器启动!");
        while (true) {
            // 通过accept方法来”接听电话“,然后才能进行
            Socket clientSocket = serverSocket.accept();
            // 和客户端的交互
            processConnection(clientSocket);
        }
    }

    // 通过这个方法开处理一次连接,连接建立的过程中就会涉及到多次的请求响应交互
    private void processConnection(Socket clientSocket) {
        System.out.printf("[%s:%d] 客户端上线!\n",clientSocket.getInetAddress(),clientSocket.getPort());
        // 循环的读取客户端的请求并返回响应
        // 从网卡读/写数据,tcp是面向字节流和文件中的字节流完全一致
        try (InputStream inputStream = clientSocket.getInputStream();
             OutputStream outputStream = clientSocket.getOutputStream()) {
            while (true) {
                Scanner scanner = new Scanner(inputStream);
                if (!scanner.hasNext()) {
                    // 读取完毕,客户端断开连接,就会产生读取完毕
                    System.out.printf("[%s:%d] 客户端下线!\n",clientSocket.getInetAddress(),
                            clientSocket.getPort());
                    break;
                }
                // 1. 读取请求并解析,这里注意隐的约定,next读的时候要读到空白符才会结束
                //  因此就要求客户端发来的请求必须带有空白符结尾,比如 \n 或空格
                String request = scanner.next();
                // 2. 根据请求计算响应
                String response = process(request);
                // 3. 把响应返回给客户端
                //  通过这种方式可以写回,但是这种方式不方便给返回的响应添加 \n
                //outputStream.write(response.getBytes(),0,response.getBytes().length);
                // 也可以把outputStream 套上一层,完成更方便的写入
                PrintWriter printWriter = new PrintWriter(outputStream);
                printWriter.println(response);
                // PrintWriter 内置的缓冲区需要冲刷(刷新)
                // 冲刷缓冲区
                printWriter.flush();

                System.out.printf("[%s:%d] req:%s,resp: %s\n",clientSocket.getInetAddress(),
                        clientSocket.getPort(),request,response);
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }

    public String process(String request) {
        return request;
    }

    public static void main(String[] args) throws IOException {
        TcpEchoServer tcpEchoServer = new TcpEchoServer(9090);
        tcpEchoServer.start();
    }
}

3.2 TCP客户端

// Socket 对应到网卡,既可以给服务器使用,又可以给客户端使用
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;

public class TcpEchoClient {
    private Socket socket = null;
    public TcpEchoClient(String serverIP,int serverPort) throws IOException {
        // 此处可以把这里的ip和port直接传给socket对象
        // 由于tcp是有连接的,因此socket里面就会保存好这俩信息,因此此处TcpEchoClient类就不必保存
        socket = new Socket(serverIP,serverPort);
    }

    public void start() {
        System.out.println("客户端启动!\n");
        try (InputStream inputStream = socket.getInputStream();
             OutputStream outputStream = socket.getOutputStream()) {
            Scanner scannerConsole = new Scanner(System.in);
            Scanner scannerNetwork = new Scanner(inputStream);
            PrintWriter writer = new PrintWriter(outputStream);
            while (true) {
                // 1. 从控制台读取输入的字符串
                System.out.print("-> ");
                if (!scannerConsole.hasNext()) {
                    // 没有下一个
                    break;
                }
                // 有下一个接着读取
                String request = scannerConsole.next();
                // 2. 把请求发给服务器  使用println 来发送的请求末尾带有\n
                //  这里是和服务器的scanner.next 呼应的
                writer.println(request);
                // 通过flush 主动刷新缓冲区,确保数据真的发出去了
                writer.flush();
                // 3.从服务器读取响应
                String response = scannerNetwork.next();
                // 4. 把响应显示出来
                System.out.println(response);
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }


    public static void main(String[] args) throws IOException {
        TcpEchoClient client = new TcpEchoClient("127.0.0.1",9090);
        client.start();
    }
}

3.3 整个流程

  1. 服务器启动,阻塞在accept,等待客户端连接
  2. 客户端启动,new Socket(ServerIP,ServerPort)操作会触发和服务器之间的建立连接的操作,此时服务器就会从accept中返回
  3. 服务器从accept返回,进入到processConnection方法,执行到hasNext这里,产生阻塞,此时虽然连接建立了,但是客户端还没有发送任何的请求,hasNext阻塞等待请求到达。
  4. 客户端继续执行到hasNext,等待用户向控制台写入内容。
  5. 用户输入之后,此时hasNext就返回了,继续执行这里的发送请求的逻辑,这里就会把请求真的发出去,同时客户端等待服务器的响应返回,next也会产生阻塞。
  6. 服务器从hasNext返回读取到请求内容进行处理,读取到请求,构造出响应,把响应写回到客户端。服务器结束这次循环,继续阻塞在hasNext等待下一个请求。
  7. 客户端读取响应,并且显示出来。结束这次循环,进行下一次循环,继续阻塞在hasNext等待用户输入第二个数据。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/484831.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Django 三板斧、静态文件、request方法

【一】三板斧 【1】HttpResponse (1)介绍 HttpResponse是Django中的一个类,用于构建HTTP响应对象。它允许创建并返回包含特定内容的HTTP响应。 (2)使用 导入HttpResponse类 from django.http import HttpResponse创…

C++ unordered_set和unordered_map

哈希 1. unordered_set/unordered_map1.1 背景1.2 unordered_set1.2.1 特性1.2.2 常用方法 1.3 unordered_map1.3.1 特性1.3.2 常用方法 2. 哈希2.1概念2.2 哈希冲突2.2.1哈希函数2.2.2 解决哈希冲突2.2.2.1 闭散列2.2.2.2 开散列 1. unordered_set/unordered_map 1.1 背景 之…

Rust并发编程thread多线程和channel消息传递

安全高效的处理并发是 Rust 诞生的目的之一,主要解决的是服务器高负载承受能力。 并发(concurrent)的概念是指程序不同的部分独立执行,这与并行(parallel)的概念容易混淆,并行强调的是"同…

如何理解OSI七层模型?

一、是什么 OSI (Open System Interconnect)模型全称为开放式通信系统互连参考模型,是国际标准化组织 ( ISO ) 提出的一个试图使各种计算机在世界范围内互连为网络的标准框架 OSI将计算机网络体系结构划分为七层,每一层实现各自…

存储随笔原创科普视频首播~

一周之前,存储随笔创建了B站账号。小编利用上个周末休息时间专门研究了B站视频录制的各种方案。发现并没有想象的很容易,先花了很长时间准备了一个PPT,再准备演讲大纲,最终磕磕绊绊完成了首期原创视频录制! 可能不尽如…

PCB布线中晶振电容、电源大小电容、电源电容的设计细节

嵌入式软硬件爱好者 一张手册走天下。嵌入式单片机/Linux/Openwrt/电子电路技术交流分享。//主打一个技术层面的剑走偏锋,直击众人重视和不重视的重点//专注基础,才能走的更远 晶振电容 晶振旁边的电容在电路设计中不是用于滤波的。实际上,…

中国疆域从古至今版图演变,中国历史各个朝代地图大全

一、图片描述 每个朝代都有数十张地图,朝代疆域全图重点区域地图,图片是JPG格式,都是高清地图,行政名称清晰可见,非常适合喜欢历史的朋友。本套历史朝代地图,大小1.32G,18个压缩文件。 二、图…

ShardingSphere水平分表——开发经验(2)

1. 什么场景下分表? 数据量过大或者数据库表对应的磁盘文件过大。 Q:多少数据分表? A:网上有人说1kw,2kw?不准确。 1、一般看字段的数量,有没有包含text类型的字段。我们的主表里面是不允许有t…

C语言数据结构之归并排序

疏雨池塘见 微风襟袖知 目录 归并排序的介绍 基本思想 时间复杂度分析 ⭐归并排序步骤 空间复杂度分析 代码展示 ✨归并排序的非递归 代码展示 总结🔥 归并排序的介绍 归并排序,是创建在归并操作上的一种有效的排序算法。算法是采用分治法&#xff…

项目1-加法计算器

1.创建项目 2.导入前端代码 2.1 static包内 2.2 测试前端代码是否有误 显示成功说明无误 2.3 定义用户接口 请求路径:calc/sum 请求方式:GET/POST 接口描述:计算两个整数相加 请求参数: 参数名类型是否必须备注num1Integer是参与计算的第…

瑞萨杯(一)

基础信息 RA6M5:ARM V8架构,24MHz外置晶振,200MHz主频 SCI(Serial Communications Interface),意为串行通信接口 参考链接: 【瑞萨RA系列FSP库开发】RASCKeil的环境搭建_瑞萨ra mdk-CSDN博客…

主干网络篇 | YOLOv8改进之在主干网络中引入密集连接卷积网络DenseNet

前言:Hello大家好,我是小哥谈。DenseNet(密集连接卷积网络)是一种深度学习神经网络架构,它在2017年由Gao Huang等人提出。DenseNet的核心思想是通过密集连接(dense connection)来促进信息的流动和共享。在传统的卷积神经网络中,每个层的输入只来自于前一层的输出。而在…

C语言之strsep用法实例(八十六)

简介: CSDN博客专家,专注Android/Linux系统,分享多mic语音方案、音视频、编解码等技术,与大家一起成长! 优质专栏:Audio工程师进阶系列【原创干货持续更新中……】🚀 优质专栏:多媒…

【Python音视频技术】Python音视频技术系列文章2---视频提取音频转换文字

接上一篇文章 【Python音视频技术】玩AI视频创作引发写Python音视频技术系列文章1---视频添加字幕 之前我想在视频中提取音频转换文字, 当时是用 PC剪映专业版搞定的, 详情见 【AI应用】模仿爆款视频二次创作短视频操作步骤 。 这里我准备用pytho…

铁道障碍物检测6种YOLOV8

铁道障碍物检测6种,采用YOLOV8训练,得到PT模型,然后转换成ONNX模型,OPENCV调用 铁道障碍物检测6种YOLOV8

android Fragment 生命周期 方法调用顺序

文章目录 Introlog 及结论代码 Intro 界面设计:点击左侧按钮,会将右侧 青色的RightFragment 替换成 黄色的AnotherRightFragment,而这两个 Fragment 的生命周期方法都会打印日志。 所以只要看执行结果中的日志,就可以知道 Fragme…

Linux 系统 快速卸载docker

(卸载前一定要做好相关数据的备份) 卸载: 第一种卸载方法 1、查询docker安装过的包: yum list installed | grep docker 2、删除安装包: yum remove docker-ce.x86_64 ddocker-ce-cli.x86_64 -y 3、删除镜像/容器等 rm -rf /var/lib/dock…

IT运维服务规范标准与实施细则

一、 总则 本部分规定了 IT 运维服务支撑系统的应用需求,包括 IT 运维服务模型与模式、 IT 运维服务管理体系、以及 IT 运维服务和管理能力评估与提升途径。 二、 参考标准 下列文件中的条款通过本部分的引用而成为本部分的条款。凡是注日期的引用文件&#xff0c…

基于QT的实现的人脸识别、人脸标记、人脸比对

该项目使用的人脸模型框架采用的是seetaface开源版本,经过测试发现效果还算可以。 人脸识别的效果图如下: 人脸比对的效果图如下: 鉴于测试识别的精度特意找了不同两人相似的人脸进行比对,效果如下图: 由于该模型采用的阈值是0.6…

前端框架前置课(1)---AJAX阶段

1. AJAX入门 1.1 AJAX概念和axios使用 1.1.1 什么是AJAX? 1.1.2 怎么用AJAX? 引入axios.js 获取省份列表数据 1.2 认识URL 1.3 URL查询参数 1.4 常用请求方和数据提交 1.5 HTTP协议-报文 1.5.1 HTTP响应状态码 1.5.1.1 状态码:1XX(信息&#xff09…
最新文章