JavaEE|套接字编程之UDP数据报

news2024/11/15 15:32:10

文章目录

    • 一、DatagramSocket API
      • 构造方法
      • 常用方法
    • 二、DatagramPacket API
      • 构造方法
      • 常用方法
    • E1:回显服务器的实现
    • E2:带有业务逻辑的请求发送

一、DatagramSocket API

在操作系统中,把socket对象当成了一个文件处理。等价于是文件描述符表上的一项。

普通的文件,对应的硬件设备是硬盘,而socket文件,对应的硬件设备是网卡。

【网卡:是一块被设计用来允许计算机在计算机网络上进行通讯的计算机硬件。】

这也体现了计算机资源一切皆文件的理念。

如果要和多个不同的主机进行通信,就需要创建多个socket对象。

构造方法

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

port是一个端口号,绑定了接收方进程。

那不指定端口号的构造方法呢?此时系统会自动分配一个空闲的端口。

本质上不是进程和端口号建立了联系,而是进程中的socket对象和端口建立了联系。

常用方法

方法签名方法说明
void receieve(DatagramSocket p)从当前套接字对象接受数据报。如果没有接收到会阻塞等待【输出形参数】
void send(DatagramSocket p)从当前套接字对象发送数据包。不会阻塞等待,直接发送
void close()关闭这个数据报套接字对象【类比文件对象关闭,释放资源】

二、DatagramPacket API

它是表示udp中传输的一个报文,构造这个对象,可以指定一些具体的数据进去。

构造方法

方法签名说明
DatagramPacket(byte[] buf,int length)构造接收数据报的对象,接受的数据保存在字节数组(buf)中,接收指定长度(len)
DatagramPacket(byte[] buf,int offset,int length,SocketAddress address)构造一个DatagramPacket以用来发送数据报的对象,发送的数据为字节数组(buf里),从0到指定长度(length)。address指定目的主机ip和端口号 【SocketAddress这个类表示ip+端口号】

常用方法

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

可以用一个线程receive,把数据放进阻塞队列中,另一个线程进行处理请求并响应。

手定指定,并不一定数组全部用,但是指定长度全部用,留微操空间。不过现在用的比较少了。

E1:回显服务器的实现

普通服务器和回响服务器的区别:

普通服务器,根据收到的请求个性化的返回对应的响应。

回显服务器,省略了其中的根据请求计算响应,请求是什么,就返回什么。

后者代码里边没有实际业务。而实际上最关键的是根据请求计算响应的环节。

注意事项:

  1. 端口号的设置:服务器可以显示指定,是因为是程序员手中的,是可控的,但是客户端一般是用户的,是不可控的E2:请求响应
  2. 个别参数的指定:①respose.length_字符的个数;respose.getBytes().length是字节的个数②offset是一个偏移量,相当于起始位置

UDP回响服务器代码

public class Code01UDPEchoServer {
    private DatagramSocket socket=null;//OS提供操作网卡的socket对象
    //服务器一定要关联上一个端口号!!!不能让它随机分配,方便端口号定位
    public Code01UDPEchoServer(int port) throws SocketException {
        socket=new DatagramSocket(port);
    }

    public void start() throws IOException {
        System.out.println("服务器启动!!");

//        //存取数据
//        byte[] bytes=new byte[4096];//2^12
//      这里是直接每次new 了
//        int length=0;


        //因为要服务很多客户端,所以服务器需要一直在就绪状态
        while (true) {
            //1.获取客户端发送过来的请求
            //  receive是一个输出型参数,所以我们需要先构造一个DatagramPacket的对象
            DatagramPacket requestPacket = new DatagramPacket(new byte[4096], 4096);//类似给人家一个空白的纸条,两个人说悄悄话
            socket.receive(requestPacket);

            //由于这里是一个特殊对象,所以一般把它拿出来做成字符串
            //offset是一个偏移量
            //这里指定范围,节省构造字符串的空间[指构造有效范围的]
            String request = new String(requestPacket.getData(), 0, requestPacket.getLength());

            //2.根据请求计算响应(请求与响应相同)
            String respose=process(request);

            //3.把响应写回客户端
            //将响应字符串转成字节数组

            //respose.length_字符的个数
            //respose.getBytes().length是字节的个数

            //怎么确定响应返回端口——通过getSocketAddress
            DatagramPacket answerPacket=new DatagramPacket(respose.getBytes(),respose.getBytes().length,requestPacket.getSocketAddress());
            socket.send(answerPacket);

            //4.打印本次请求的处理中间结果
            System.out.printf("[%s:%d] req:%s\n",requestPacket.getAddress().toString(),
                    requestPacket.getPort(),request,respose);//获得ip和端口
        }
    }
    //根据请求计算响应
    public String process(String request){
        return request;
    }

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

UDP回响客户端

public class Code02UDPEchoClient {
    private DatagramSocket socket=null;
    public Code02UDPEchoClient(String serverip, int serverport) throws SocketException {
        socket=new DatagramSocket();//不需要显示绑定端口,os随机分配
        serverIp=serverip;
        serverPort=serverport;
    }
    private String serverIp=null;
    private Integer serverPort=null;
    //一次通信,需要有两个ip,两个端口,客户端的ip是环回ip,端口号是操作系统随机分配的;服务器的ip和端口号需要声明
    public void start() throws IOException {
        System.out.println("客户端启动!");
        Scanner scanner = new Scanner(System.in);
        while (true) {
            // 1. 从控制台读取要发送的数据
            System.out.print("> ");
            String request = scanner.next();
            if (request.equals("exit")) {
                System.out.println("bye");
                break;
            }
            // 2. 构造成 UDP 请求, 并发送
            //    构造这个 Packet 的时候, 需要把 serverIp 和 port 都传入过来. 但是此处 IP 地址需要填写的是一个 32位的整数形式.
            // 这里的 IP 地址是一个字符串. 需要使用 InetAddress.getByName 来进行一个转换.端口号一直都是integer没关系
            DatagramPacket requestPacket = new DatagramPacket(request.getBytes(), request.getBytes().length, InetAddress.getByName(serverIp), serverPort);
            socket.send(requestPacket);
            // 3. 读取服务器的 UDP 响应, 并解析
            DatagramPacket responsePacket = new DatagramPacket(new byte[4096], 4096);
            socket.receive(responsePacket);
            String response = new String(responsePacket.getData(), 0, responsePacket.getLength());
            // 4. 把解析好的结果显示出来.
            System.out.println(response);
        }
    }
    public static void main(String[] args) throws IOException {
        Code02UDPEchoClient client = new Code02UDPEchoClient("127.0.0.1", 1200);
        client.start();
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-XSqYLyLp-1676721291836)(F:\typora插图\image-20230218194828403.png)]

默认情况下,是只允许运行一个客户端,但是我们可以通过设置同时运行两个客户端的

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-refFjdqY-1676721291838)(F:\typora插图\image-20230218195123652.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-glSfyeRD-1676721291839)(F:\typora插图\image-20230218195211579.png)]

E2:带有业务逻辑的请求发送

思路:复用之前的echoServer的代码,只需要重写process的逻辑,即修改业务内容。

public class Code03_UdpDictServer extends Code01UDPEchoServer {
    private Map<String,String> map=new HashMap<>();
    public Code03_UdpDictServer(int port) throws SocketException {
        super(port);
        map.put("dog","小狗");
        map.put("cat","小猫");
        map.put("pig","小猪");
    }
    public String process(String request){
        return map.getOrDefault(request,"尚未查出结果");
    }

    public static void main(String[] args) throws IOException {
        Code03_UdpDictServer udpDictServer=new Code03_UdpDictServer(1200);
        udpDictServer.start();
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-LpoUcOGJ-1676721291840)(F:\typora插图\image-20230218194650960.png)]

当然,除了使用map,我们还可以把词存在硬盘上,通过文件io操作直接读到硬盘上,这里就需要修改udpechoserver的start方法了。

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

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

相关文章

vbs简单语法及简单案例

文章目录一、简单语法1、变量2、输入3、输出4、选择语句5、循环二、用记事本编译中文乱码问题三、制作一个简单vbs脚本表白一、简单语法 1、变量 语法&#xff1a; dim 变量名例&#xff1a; dim a,b a1 b2 msgbox ab运行&#xff1a; 2、输入 语法&#xff1a;InputBox(…

【ip neigh】管理IP邻居( 添加ARP\NDP静态记录、删除记录、查看记录)

一、邻居管理存在状态 1、NUD_NONE&#xff1a; 初始状态。当一个新的路由缓存条目被创建时&#xff0c;arp_bind_neighbour()函数被调用.如果找不到相匹配的ARP缓存条目, neigh_alloc()将创建一个新的ARP缓存条目并设置状态为NUD_NONE. 2、NUD_INCOMPLETE&#xff1a;未完成状…

设计模式之适配器模式与桥接模式详解和应用

目录1 适配器模式1.1 定义1.2 应用场景1.3 适配器角色1.4 类适配器1.5 对象适配器1.5 接口适配器1.6 实战1.7 源码1.8 适配器与装饰器的对比1.9 适配器模式的优缺点1.10 总结2 桥接模式2.1 原理解析2.2 角色2.3 通用写法2.4 应用场景2.5 业务场景中的运用2.6 源码2.7 桥接模式优…

指针笔记(指针数组和指向数组的指针,数组中a和a的区别等)

指针数组和指向数组的指针 int *p[4]和int (*p)[4]有何区别&#xff1f; 前者是一个指针数组&#xff0c;数组大小为4&#xff0c;每一个元素都是一个指向int的指针 后者是指向int[4]类型数组的指针 以上代码若运行会报如下错误 main函数中定义的a数组本质是一个指向int[2]的…

内网渗透(三十八)之横向移动篇-pass the key 密钥传递攻击(PTK)横向攻击

系列文章第一章节之基础知识篇 内网渗透(一)之基础知识-内网渗透介绍和概述 内网渗透(二)之基础知识-工作组介绍 内网渗透(三)之基础知识-域环境的介绍和优点 内网渗透(四)之基础知识-搭建域环境 内网渗透(五)之基础知识-Active Directory活动目录介绍和使用 内网渗透(六)之基…

从0到1一步一步玩转openEuler--18 openEuler 管理服务-简介

文章目录18 管理服务简介18.1 概念介绍18 管理服务简介 systemd是在Linux下&#xff0c;与SysV和LSB初始化脚本兼容的系统和服务管理器。systemd使用socket和D-Bus来开启服务&#xff0c;提供基于守护进程的按需启动策略&#xff0c;支持快照和系统状态恢复&#xff0c;维护挂…

java基础学习 day41(继承中成员变量和成员方法的访问特点,方法的重写)

继承中&#xff0c;成员变量的访问特点 a. name前什么都不加&#xff0c;name变量的访问采用就近原则&#xff0c;先在局部变量中查找&#xff0c;若没找到&#xff0c;继续在本类的成员变量中查找&#xff0c;若没找到&#xff0c;继续在直接父类的成员变量中查找&#xff0c…

Mel Frequency Cepstral Coefficients (MFCCs)

wiki里说 在声音处理中&#xff0c;梅尔频率倒谱( MFC ) 是声音的短期功率谱的表示&#xff0c;基于非线性梅尔频率标度上的对数功率谱的线性余弦变换。 倒谱和MFC 之间的区别在于&#xff0c;在 MFC 中&#xff0c;频带在梅尔尺度上等距分布&#xff0c;这比正常频谱中使用的线…

Windows10 安装ElasticStack8.6.1

一、安装ElasticSearch8.6.1 1.官网下载ElasticSearch8.6.1压缩包后解压 2.安装为服务 elasticsearch-service.bat install 3.运行 elasticsearch-service.bat start 4.通过浏览器访问 http://localhost:9200/ 提示需要登录&#xff0c;但不知密码是啥。 5.重置密码 ela…

操作系统(day12)-- 基本分段存储,段页式存储

基本分段存储管理方式 不会产生内部碎片&#xff0c;会产生外部碎片 分段 按照程序自身的逻辑关系划分为 若干个段&#xff0c;每个段都有一个段名&#xff0c;每段从0开始编址 分段存储管理方式中一个段表项由段号&#xff08;隐含&#xff09;、段长、基地址 分段的段表项固…

Python基础2

1. python函数定义 函数定义语法&#xff1a; def 函数名&#xff08;传入参数&#xff09;&#xff1a; 函数体 return 返回值 —————————————— 参数如果不需要&#xff0c;可以省略返回值如果不需要&#xff0c;可以省略函数必须先定义在使用 注意&#xff…

UIE微调:autoML平台实践(一)

参考&#xff1a;uie模型微调个人总结 github&#xff1a;https://github.com/PaddlePaddle/PaddleNLP/tree/develop/model_zoo/uie 1、显存问题 之前是在实验室的服务器上跑&#xff08;2080ti&#xff0c;12G显存&#xff09;&#xff0c;频频出现以下报错&#xff1a; 原…

python 刷题时常见的函数

collections.OrderedDict 1. move_to_end() move_to_end() 函数可以将指定的键值对移动到最前面或者最后面&#xff0c;即最左边或最右边 。 2. popitem() popitem()可以完成元素的删除操作&#xff0c;有一个可选参数last&#xff08;默认为True&#xff09;&#xff0c;…

微软?还是Linux?

导读本周&#xff0c;微软宣布它成为了 Linux 基金会的白金成员&#xff0c;这距其前 CEO 巴尔默将 Linux 称之为“癌症”才 15 年。虽然此举对微软来说意义重大&#xff0c;但是并不是开源界的每个人都认为这对于 Linux 来说是好的变化&#xff0c;特别是这家位于雷德蒙的软件…

kafka-3-kafka应用的核心要点和内外网访问

kafka实战教程(python操作kafka)&#xff0c;kafka配置文件详解 Kafka内外网访问的设置 1 kafka简介 根据官网的介绍&#xff0c;ApacheKafka是一个分布式流媒体平台&#xff0c;它主要有3种功能&#xff1a; (1)发布和订阅消息流&#xff0c;这个功能类似于消息队列&#x…

宁波大学2023年MBA招生考试初试成绩查询的通知

根据往年的情况&#xff0c;2023宁波大学MBA考试初试成绩可能将于2月21日公布&#xff0c;最早于20号出来&#xff0c;为了广大考生可以及时查询到自己的分数&#xff0c;杭州达立易考教育为大家汇总了信息。 宁波大学2023年全国硕士研究生招生考试初试成绩将于2月21日公…

Java Type类

文章目录Type简介Type分类1. 原始类型(Class)2. 参数化类型(ParameterizedType)3. 类型变量(TypeVariable)4. 通配符类型(WildcardType)5. 泛型数组类型(GenericArrayType)Type简介 Type是Java编程语言中所有类型的公共高级接口。它们包括原始类型、参数化类型、数组类型、类型…

Java数据结构-栈、队列常用类(Stack、ArrayDeque、LinkedLList)

数据结构的三要素包括&#xff1a;逻辑结构、存储结构、数据的运算。逻辑结构描述的是数据之间的逻辑关系&#xff0c;分为线性结构&#xff08;线性表&#xff08;数组、链表&#xff09;、栈、队列&#xff09;和非线性结构&#xff08;图、树、集合&#xff09;。物理结构也…

服务器是干什么用的?

首先&#xff0c;什么是服务器&#xff1f;服务器是提供计算服务器和网络服务的设备。服务器和计算机由CPU、硬盘、内存、系统总线等组成。比如我们访问一个网站&#xff0c;点击这个网站会发出访问请求&#xff0c;服务器会响应服务请求&#xff0c;进行相应的处理&#xff0c…

在windows中使用tomcat搭建Jenkins

1、 准备环境&#xff1a;JDK JDK官网下载&#xff1a;https://download.oracle.com/java/19/latest/jdk-19_windows-x64_bin.msi 2、 tomcat包 tocat官网下载&#xff1a;https://tomcat.apache.org/download-90.cgi 3、 Jenkins.war包 Jenkins官网下载&#xff1a;https://mi…