UDP数据报套接字

news2024/11/28 14:36:18

文章目录

    • DatagramSocket API
    • DatagramPacket API
    • 示例一: 请求响应
      • UDP服务端
      • UDP客户端

DatagramSocket API

Socket是操作系统中的一个概念,本质上是一种特殊的文件,Socket就属于把“网卡”这个设备给抽象成了文件。往 Socket 文件中写数据,就相当于通过网卡发数据,从 Socket 文件中读数据,就相当于通过网卡接收数据。而在Java中,就使用 DatagramSocket 这个类来表示系统内部的 Socket 文件。

DatagramSocket 构造方法

方法方法说明
DatagramSocket()创建一个UDP数据报的Socket,绑定到本机任意一个随机端口(一般用于客户端)
DatagramSocket(int port)创建一个UDP数据报的Socket,绑定到本机指定的端口(一般用于服务端)

DatagramSocket 方法:

方法方法说明
void receive(DatagramPacket p)使用DatagramPacket这个类来表示一个UDP数据报(如果没有接收到数据报,该方法会阻塞等待)
void send(DatagramPacket p)从此套接字中发送数据报(不会阻塞等待,直接发送)
void close()关闭数据报套接字

DatagramPacket API

DatagramPacket 构造方法:

方法方法说明
DatagramPacket(byte[] buf, int length)构造一个DatagramPacket 用来接收数据报,接收的数据保存在字节数组中,指定接收长度(第二个参数length)
DatagramPacket(byte[] buf, int offset, int length, SocketAddress address)构造一个DatagramPacket 用来发送数据报,发送的数据为字节数组(第一个参数buf)中,从0到指定长度(第二个参数 length),address指定目的主机的IP和端口号

DatagramPacket 方法:

方法方法说明
InetAddress getAddress()从接收的数据报中,获取发送端主机IP地址;或从发送的数据报中,获取接收端主机IP地址
int port()从接收的数据报中,获取发送端主机的端口号;或从发送的数据报中,获取接收端主句端口号
byte[] getData()获取数据报中的数据

示例一: 请求响应

任务需求:

1.在客户端这边,从键盘上输入一个路径

2.发送请求,将这个路径构造成数据报发送给服务器

3.服务器接收到请求并计算响应:根据该请求数据,如果是一目录,列出目录中所包含的所有文件及文件夹,如果是一个普通文件,列出文件所在目录中的所有文件及文件夹

4.服务端返回响应:遍历子文件和子文件夹,每个文件名一行,作为响应的数据报返回给客户端

5.客户端接收响应:打印出所有的文件及文件夹

注意:为了解决空字符或读取请求时读取的字符串长度不够,造成数据丢失的情况,这里约定,客户端和服务端发送的数据都已 ‘/3’ 进行结尾,读取到 ‘/3’ 就表示请求或响应都全部读取完成

UDP服务端

public class UdpEchoServer {
    //1.创建一个DatagramSocket,用于后续接收和发送UDP数据报
    private static DatagramSocket socket;
    private static DatagramPacket requestPacket;
    public UdpEchoServer(int port) throws SocketException {
        //在服务器这边使用Socket,一般要显示的指定端口号
        //在客户端,不需要显示的指定端口号,系统会自动分配一个端口号
        socket = new DatagramSocket(port);
    }
    public static void start() throws IOException {
        while(true) {
            //2.创建数据报,用于接收客户端发来的数据报
            byte[] dataRequest = new byte[4024];
            requestPacket = new DatagramPacket(dataRequest, dataRequest.length);
            System.out.println("----------------------------------------");
            System.out.println("等待服务端接收数据");

            //3.等待客户端发来的数据报,在服务端收到数据报之前,receive()方法会一直阻塞等待,直到收到数据报后,
            // DatagramPacket对象中,包含客户端发来的数据,客户端IP地址和端口号
            socket.receive(requestPacket);
            System.out.println("客户端IP:" + requestPacket.getAddress().getHostAddress());
            System.out.println("客户端端口号:" + requestPacket.getPort());

            //7. 根据请求,计算响应
            for(int i = 0; i < dataRequest.length; i++) {
                byte b = dataRequest[i];
                if(b == '\3') {
                    //7.1 读取请求:读取到约定好的结束符'\3',取结束符前的所有内容
                    String request = new String(dataRequest, 0 , i);
                    System.out.println("请求的路径:" + request);
                    //7.2 根据请求计算响应
                    File requestFile = new File(request);
                    File[] children = null;
                    //7.3 判断该路径是否存在
                    if(requestFile.exists()) {
                        if(!requestFile.isDirectory()) {
                            //表示该对象是一个普通文件
                            //获取上级目录路径
                            String parent = requestFile.getParent();
                            requestFile = new File(parent);
                        }
                        children = requestFile.listFiles();
                    }else {
                        String res = "该路径错误!!!" +'\n' + '\3';
                        sendEcho(res);
                        break;
                    }
                    //7.4 构造响应内容,将每个子文件作为一行
                    StringBuilder response = new StringBuilder();
                    for(File child : children) {
                        response.append(child.getName() + '\n');
                    }
                    //7.5 读取完之后,加上约定的结束符
                    response.append('\3');
                    //7.6 发送返回的响应数据
                    sendEcho(response.toString());
                    break;
                }
            }
        }
    }
    public static void sendEcho(String response) throws IOException {
        //构造响应数据报
        byte[] responseData = response.getBytes(StandardCharsets.UTF_8);
        DatagramPacket responsePacket = new DatagramPacket(responseData, responseData.length, requestPacket.getSocketAddress());
        //发送返回的响应数据报
        socket.send(responsePacket);
        System.out.println("发送完成");
    }
    public static void main(String[] args) {
        try {
            UdpEchoServer echoServer = new UdpEchoServer(8888);
        } catch (SocketException e) {
            throw new RuntimeException(e);
        }
        try {
            UdpEchoServer.start();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }
}

UDP客户端

public class UdpEchoClient {

    //4.创建客服端Socket
    //4.1指定服务端端地址和端口号
    private static final SocketAddress ADDRESS = new InetSocketAddress("127.0.0.1", 8888);
    private static DatagramSocket socket;
    public UdpEchoClient() {
        //客户端不需要指定端口号
        try {
            socket = new DatagramSocket();
        } catch (SocketException e) {
            throw new RuntimeException(e);
        }
    }
    public static void start() throws IOException {


        Scanner in = new Scanner(System.in);
        while(true) {

            //5.构建数据报
            System.out.println("请输入要展开的目录");
            //5.1 为了接收端能获取到有效的内容,以\3作为结束符;
            String request = in.nextLine() + '\3';
            //5.2 将要发送的数据转换成字节,并指定字符集
            byte[] requestData = request.getBytes(StandardCharsets.UTF_8);
            //5.3 组装好的数据包中包含了数据,及发送服务端的信息(IP地址、端口号)
            DatagramPacket requestPacket = new DatagramPacket(requestData, requestData.length, ADDRESS);

            //6.发送数据报
            socket.send(requestPacket);
            System.out.println("--------------------------------");

            //8.接收服务端响应的数据报,打印出响应内容
            //8.1 使用字节数组接收响应数据
            byte[] responseData = new byte[4090];
            //8.2 构建响应数据报
            DatagramPacket responsePacket = new DatagramPacket(responseData, responseData.length);
            //8.3 接收响应
            socket.receive(responsePacket);
            //8.4 打印出响应内容
            System.out.println("该目录下的文件列表为:");
            //因为,响应中的每个文件都是以回车结尾的,next用于定义每个文件起始位置,
            int next = 0;
            for(int i = 0; i < responseData.length; i++) {
                byte b = responseData[i];
                if(b == '\3') {
                    //表示响应读取完成
                    break;
                }
                if(b == '\n') {
                    //遇见回车键表示读取到了一个文件名
                    //构建字符串文件名
                    String print = new String(responseData, next, i - next);
                    System.out.println(print);
                    //下次解析从换行后的索引开始
                    next = i + 1;
                }
            }
        }

    }

    public static void main(String[] args) {
        UdpEchoClient echoClient = new UdpEchoClient();
        try {
            UdpEchoClient.start();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }
}

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Qt 中的窗口类

目录 QWidget 设置父对象 窗口位置 测试代码 窗口尺寸 窗口标题和图标 信号 槽函数 QDialog 常用API QDialog的子类 QMessageBox QFileDialog QFontDialog QFontDialog类的静态API QColorDialog 颜色类 QColor 静态API函数 测试代码 QInputDialog 静态函数…

花店小程序商城制作攻略教程分享

现如今&#xff0c;随着互联网的快速发展&#xff0c;越来越多的实体店面对客流量不足的问题。特别是对于花店来说&#xff0c;客流量的多少直接影响着销售额和收益。为了解决这一问题&#xff0c;开发一个花店小程序商城成为了不可忽视的选择。 为了开发花店小程序商城&#x…

10倍提升启动的时间?Graalvm打包Springboot+MyBatis实测

graalvm使用前后对比图 相关代码博客&#xff1a;https://blog.csdn.net/weixin_43914278/article/details/134446327 工具大小时间graalvm打包的exe文件84.14MB0.251秒graalvm打包的docker文件121.27MB0.253秒jar包51.34MB2.153秒 解析 文件大小: graalvm打包的Docker文件…

【数电笔记】09-逻辑代数的基本定律、常用公式

目录 说明&#xff1a; 逻辑代数的基本定律 1. 常量间的运算 2. 逻辑变量与常量的运算 3. 与普通代数相似的定律 4. 摩根定律&#xff08;反演律&#xff09; 5. 等式证明方法例题 逻辑代数的常用公式 1. 吸收律 2. 冗余律 3. 示例应用 4. 关于异或运算的一些公式 …

【深度学习笔记】08 欠拟合和过拟合

08 欠拟合和过拟合 生成数据集对模型进行训练和测试三阶多项式函数拟合&#xff08;正常&#xff09;线性函数拟合&#xff08;欠拟合&#xff09;高阶多项式函数拟合&#xff08;过拟合&#xff09; import math import numpy as np import torch from torch import nn from d…

动态类型语言与静态类型语言的对比与比较

编程语言可以根据类型系统和类型检查时机分为动态编程语言和静态编程语言两大类&#xff0c;它们在运行时的代码检查方式、变量类型的使用方式等方面有很大的区别。这一块你知道吗&#xff1f; 本文将为您详细讲解两种编程语言的优缺点&#xff0c;以及它们的应用场景。 动态编…

洗地机好用吗?口碑好的洗地机有哪些?

自从洗地机开始引入市场以来&#xff0c;它一直受到人们的关注。它在解放家庭清洁劳动力和提供快速方便的清洁方面表现出色&#xff0c;超越了多年来传统的拖把清洁方式。越来越多的人选择使用洗地机来完成家庭清洁任务。如果你也对洗地机产生了浓厚的兴趣&#xff0c;并想购买…

【精选】ATKCK红队评估实战靶场二 (超详细过程思路)

&#x1f36c; 博主介绍&#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 hacker-routing &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【应急响应】 【ATK&CK红队评估实战靶场】 【VulnHub靶场复现】【面试分析】 &#x1f…

智能优化算法应用:基于人工电场算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于人工电场算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于人工电场算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.人工电场算法4.实验参数设定5.算法结果6.参考…

IO / day03 作业

1. 使用文件IO完成对图像的读写操作 代码 #include<myhead.h>int main(int argc, const char *argv[]) {int fd-1;if((fd open("./bird.bmp", O_RDWR)) -1){perror("fopen error");return -1;}//读取该图片的大小&#xff0c;需要将光标向后偏移…

写论文焦虑?No,免费AI写作大师来帮你

先来看1分钟的视频&#xff0c;对于要写论文的你来说&#xff0c;绝对有所值&#xff01; 还在为写论文焦虑&#xff1f;免费AI写作大师来帮你三步搞定 第一步&#xff1a;输入关键信息 第二步&#xff1a;生成大纲 稍等片刻后&#xff0c;专业大纲生成&#xff08;由于举例&am…

python 堆与栈

【一】堆与栈 【 1 】简介 栈&#xff08;stack&#xff09;&#xff0c;有些地方称为堆栈&#xff0c;是一种容器&#xff0c;可存入数据元素、访问元素、删除元素&#xff0c;它的特点在于只能允许在容器的一端&#xff08;称为栈顶端指标&#xff0c;英语&#xff1a;top&a…

智能无人售货奶柜,引领便捷时代

智能无人售货奶柜&#xff0c;引领便捷时代 在当今快节奏的生活中&#xff0c;人们对购物的需求变得越来越迫切。同时&#xff0c;随着科技的进步&#xff0c;无人售货柜作为一种创新的销售模式逐渐受到人们的关注和喜爱。其中&#xff0c;智能无人售货奶柜以其便捷高效的特点成…

OGG实现Oracle19C到postgreSQL14的实时同步

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是【IT邦德】&#xff0c;江湖人称jeames007&#xff0c;10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】&#xff01;&#x1f61c;&am…

TP5使用Composer安装phpoffice/phpspreadsheet,导出Excel文件

Composer安装 如果你尚未安装Composer&#xff0c;请先安装 Composer。Composer是PHP的依赖管理工具&#xff0c;它可以方便地安装和管理项目中的第三方库。 安装phpoffice/phpspreadsheet&#xff1a; 触发控制器里面的方法 wdjzdc() 在控制中引入 use PhpOffice\PhpSpread…

从零开始的Spring Cloud Gateway指南:构建强大微服务架构

目录 一、 什么是Gateway&#xff1f;1. 网关的由来2. 网关的作用3. 网关的技术实现 二、如何搭建一个简易网关服务1. 引入依赖2. 配置yml文件 三、进阶话题&#xff1a;过滤器和路由配置1. gateway的执行原理2. 路由断言工厂: Predicate Factory3. 网关过滤器&#xff1a;Gate…

Cannot resolve symbol ‘ActivityResultLauncher‘ 报错处理方法

修改 app/build.gradle implementation ‘androidx.appcompat:appcompat:1.2.0’ 为 implementation ‘androidx.appcompat:appcompat:1.4.0’

【模型报错记录】‘PromptForGeneration‘ object has no attribute ‘can_generate‘

通过这个连接中的方法解决&#xff1a; “PromptForGeneration”对象没有属性“can_generate” 期刊 #277 thunlp/OpenPrompt GitHub的 问题描述&#xff1a;在使用model.generate() 的时候报错&#xff1a;PromptForGeneration object has no attribute can_generate 解决方法…

在Spring Cloud中使用组件Zuul网关,并注册到Eureka中去

在上一篇中&#xff0c;我们搭建了Spring Cloud的父子模块&#xff0c;并实现了一个Eureka子模块的启动&#xff0c;可以通过浏览器地址去访问Eureka主页了&#xff0c;相信了解过的童鞋应该看到&#xff0c;主页上并未有任何服务去注册&#xff0c;那么我们就在这篇&#xff0…

【1】一文读懂PyQt简介和环境搭建

目录 1. PyQt简介 1.1. Qt 1.2. PyQt 1.3. 关于PyQt和PySide 2. 通过pip安装PyQt5 3. 无法运行处理 4. VSCode配置PYQT插件 PyQt官网:Riverbank Computing | Introduction 1. PyQt简介 PyQt是一套Python的GUI开发框架,即图形用户界面开发框架. Python中经常使用的GUI…