【JavaSE】【网络编程】UDP数据报套接字编程

目录

  • 一、网络编程简介
  • 二、Socket套接字
  • 三、TCP/UDP简介
    • 3.1 有连接 vs 无连接
    • 3.2 可靠传输 vs 不可靠传输
    • 3.3 面向字节流 vs 面向数据报
    • 3.4 双向工 vs 单行工
  • 四、UDP数据报套接字编程
    • 4.1 API介绍
      • 4.1.1 DatagramSocket类
        • 4.1.1.1 构造方法
        • 4.1.1.2 主要方法
      • 4.1.2 DatagramPocket类
        • 4.1.2.1 构造方法
        • 4.1.2.2 主要方法
      • 4.1.3 使用上述类创建回显服务器-客户端
        • 4.1.3.1 回显服务器UdpEchoServer
        • 4.1.3.2 回显客户端UdpEchoClient

一、网络编程简介

网络编程:

网络编程,指网络上的主机,通过不同的进程,以编程的方式实现网络通信(或称为网络数据传输)。

网络编程涉及基本概念:

一次网络数据传输时:
发送端:数据的发送方进程,称为发送端。发送端主机即网络通信中的源主机。
接收端:数据的接收方进程,称为接收端。接收端主机即网络通信中的目的主机。
收发端:发送端和接收端两端,也简称为收发端。

请求和响应 :
一般来说,获取一个网络资源,涉及到两次网络数据传输:
第一次:请求数据的发送。
第二次:响应数据的发送。

客户端和服务端
服务端:在常见的网络数据传输场景下,把提供服务的一方进程,称为服务端,可以提供对外服务。
客户端:获取服务的一方进程,称为客户端。

二、Socket套接字

Socket套接字:Socket套接字,是由系统提供用于网络通信的技术,是基于TCP/IP协议的网络通信的基本操作单元。基于Socket套接字的网络程序开发就是网络编程。

Socket套接字主要针对传输层协议划分为如下三类:

  1. 流套接字:使用传输层TCP协议。
  2. 数据报套接字:使用传输层UDP协议。
  3. 原始套接字:用于自定义传输层协议。

三、TCP/UDP简介

socket api :传输层提供给应用层的API。
传输层的两个核心协议是:TCP与UDP协议。

  • TCP协议:有连接,可靠传输,面向字节流,全双工的协议。
  • UDP协议:无连接,不可靠传输,面向数据报,全双工的协议。

3.1 有连接 vs 无连接

有/无连接:是一个抽象的概念,虚拟的逻辑上的连接。就像前面的链表的学习一样,物理上不连续,但通过节点之间的关联使其逻辑上连续。

TCP协议中:保存了对端的信息,让两个通信之间的设备有连接。就如AB通信,A保存B信息,B保存A信息。

UDP协议中:UDP本身不保存对端信息,但可以自己使用代码保存。

3.2 可靠传输 vs 不可靠传输

丢包: 网络上数据发生丢失的情况就是丢包。

传输中丢包情况造成原因:

  1. 当光/电/电磁信号,受到外界的干扰。
  2. 网络世界是通过交换机,路由器组建起来的。而交换机,路由器就像十字路口一样,当某个时间点,实际转发的数据超过设备能转发的上限的时候,就会“堵车”,造成丢包。

可靠传输:不是保证数据报100%到达,不造成一点丢失。而是尽可能提高传输成功的概率,如果丢包还会进行记录。

不可靠传输:在发出数据后就不管了。

3.3 面向字节流 vs 面向数据报

面向数据流:读写数据(读–>接收数据,写–> 发送数据)的时候,以字节为单位。支持读写任意长度,有粘包问题。
面向数据报:读写数据(读–>接收数据,写–> 发送数据)的时候,以一个数据报为单位。有长度限制。

3.4 双向工 vs 单行工

全双工:读写数据(读–>接收数据,写–> 发送数据),支持双向通信,能读能写。
单双工:面向数据流:读写数据(读–>接收数据,写–> 发送数据),只支持单向通信,要么读要么写。

四、UDP数据报套接字编程

计算机中的文件通常是一个广义的概念,文件除了指代一些我们常说的文件外,还能指代一些硬件设备。操作系统管理硬件设备也是抽象成文件。

4.1 API介绍

4.1.1 DatagramSocket类

DatagramSocket 是UDP Socket,⽤于发送和接收UDP数据报。

4.1.1.1 构造方法
方法签名方法说明
public DatagramSocket() throws SocketException创建⼀个UDP数据报套接字的Socket,绑定到本机任意⼀个随机端⼝(⼀般⽤于客⼾端)
public DatagramSocket(int port) throws SocketException创建⼀个UDP数据报套接字的Socket,绑定到本机指定的端⼝(⼀般⽤于服务端)
public DatagramSocket(int port, InetAddress laddr) throws SocketException创建⼀个UDP数据报套接字的Socket,绑定到指定的IP即laddr和端⼝port
public DatagramSocket(SocketAddress bindaddr) throws SocketException创建⼀个UDP数据报套接字的Socket,绑定到bindaddr指定的IP和端⼝
4.1.1.2 主要方法
方法签名作用
public void send(DatagramPacket p) throws IOException从此套接字发送数据报包(不会阻塞等待,直接发送)
public synchronized void receive(DatagramPacket p) throws IOException从此套接字接收数据报(如果没有接收到数据报,该⽅法会阻塞等待)
public void close()关闭此数据报套接字

4.1.2 DatagramPocket类

DatagramPacket是UDP Socket发送和接收的数据报。

4.1.2.1 构造方法
方法签名作用
public DatagramPacket(byte buf[ ], int length)构造⼀个DatagramPacket以⽤来接收数据报,接收的数据保存在字节数组(第⼀个参数buf)中,接收指定⻓度(第⼆个参数length)
public DatagramPacket(byte buf[ ], int offset, int length)构造⼀个DatagramPacket以⽤来接收数据报,接收的数据保存在字节数组(第⼀个参数buf)中,接收指定⻓度(第⼆个参数length)从buf的下标offset位置开始接收
public DatagramPacket(byte buf[], int offset, int length, SocketAddress address)构造⼀个DatagramPacket以⽤来发送数据报,发送的数据为字节数组(第⼀个参数buf)中,从offset到指定⻓度(第⼆个参数length)。address指定⽬的主机的IP和端⼝号
public DatagramPacket(byte buf[ ], int offset, int length, InetAddress address, int port)构造⼀DatagramPacket以⽤来发送数据报,发送的数据为字节数组(第⼀个参数buf)中,从0到指定⻓度(第⼆个参数length)。address指定⽬的主机的IP,port指定⽬的主机的端⼝号
public DatagramPacket(byte buf[ ], int length, SocketAddress address)构造⼀个DatagramPacket以⽤来发送数据报,发送的数据为字节数组(第⼀个参数buf)中,从0到指定⻓度(第⼆个参数length)。address指定⽬的主机的IP和端⼝号
public DatagramPacket(byte buf[ ], int length, InetAddress address, int port)构造⼀个DatagramPacket以⽤来接收数据报,接收的数据保存在字节数组(第⼀个参数buf)中,接收指定⻓度(第⼆个参数length)address指定⽬的主机的IP,port指定⽬的主机的端⼝号
4.1.2.2 主要方法
方法签名作用
public synchronized byte[ ] getData()获取数据报中的数据
public synchronized int getPort()从接收的数据报中,获取发送端主机的端⼝号;或从发送的数据报中,获取接收端主机端⼝号
public synchronized InetAddress getAddress()从接收的数据报中,获取发送端主机IP地址;或从发送的数据报中,获取接收端主机IP地址

4.1.3 使用上述类创建回显服务器-客户端

回显:请求是啥,相应就是啥。

4.1.3.1 回显服务器UdpEchoServer
  1. 先创建一个成员变量:Socket对象,用于接收发送数据报。private DatagramSocket socket = null;
  2. 构造方法,指定一个固定端口号给服务器使用,将Socket使用这个端口号实例化。
public UdpEchoServer(int port) throws SocketException{socket = new DatagramSocket(port);}
  1. 接下来在start方法中编写服务器实现逻辑。由于服务器是要7*24小时工作的,所以在一个死循环中实现。
  2. 读取请求并解析;
    4.1 构造一个DatagramPocket对象,初始化为0(不是null),用于保存UDP的载荷部分。
    DatagramPacket requestPacket = new DatagramPacket(new byte[2024],2024);
    4.2 读取请求socket.receive(requestPacket);
    4.3 把读取到有效数据的解析为字符串。
    通过getData方法获取DatagramPacket 中的字节数组,getLength方法获取DatagramPacket 有效数据长度,根据字节数组构造字符串。
    String request = new String(requestPacket.getData(), 0 ,requestPacket.getLength());
  3. 根据请求计算响应,由于是回显服务器,直接请求就是响应。
String response = product(request);private String product(String request) {return request;}
  1. 把响应返回给客户端,通过getBytes方法拿到字符串中的字节数据,getBytes().length拿到字节长度,getSocketAddress方法拿到客户端的端口和IP。
socket.send(new DatagramPacket(request.getBytes(),request.getBytes().length,
requestPacket.getSocketAddress()));
  1. 打印一个日志记录此次交互过程。
System.out.printf([%s:%d],request:%s,response:%s\n",requestPacket.getAddress().toString(),requestPacket.getPort(),request,response);
  1. 总代码
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;
public class UdpEchoServer {//先创建一个成员变量:Socket对象,用于接收发送数据报private DatagramSocket socket = null;//构造方法,指定一个固定端口号给服务器使用,将Socket使用这个端口号实例化public UdpEchoServer(int port) throws SocketException{socket = new DatagramSocket(port);}public void start() throws IOException {while (true) {//构造一个DatagramPocket对象,初始化为0(不是null),用于保存UDP的载荷部分。DatagramPacket requestPacket = new DatagramPacket(new byte[2024], 2024);//接收请求socket.receive(requestPacket);//把读取到有效数据的解析为字符串String request = new String(requestPacket.getData(),0,requestPacket.getLength());//计算响应String response = product(request);//把响应返回给客户端socket.send(new DatagramPacket(request.getBytes(),request.getBytes().length,requestPacket.getSocketAddress()));//打印一个日志System.out.printf("[%s : %d], request:%s , response:%s\n",requestPacket.getAddress().toString(),requestPacket.getPort(),request,response);}}private String product(String request) {return request;}public static void main(String[] args) throws IOException {UdpEchoServer server = new UdpEchoServer(6666);server.start();}
}
4.1.3.2 回显客户端UdpEchoClient
  1. 先创建一个成员变量:Socket对象,用于接收发送数据报private DatagramSocket socket = null;
  2. UDP 本身不保存对端的信息, 就自己的代码中保存一下.
	private String serverIp;private int port;
  1. 构造方法是要指定访问的服务器的地址。
public UdpEchoClient(String serverIp,int port) throws SocketException {this.serverIp = serverIp;this.port = port;socket = new DatagramSocket();}
  1. 从控制台读取用户输入的内容。
			System.out.println("请输入要传输的内容");Scanner scanner = new Scanner(System.in);if (!scanner.hasNext()) {break;}String request =scanner.next();
  1. 发送数据报,把请求发送给服务器, 需要构造 DatagramPacket 对象。 构造过程中, 不光要构造载荷, 还要设置服务器的 IP 和端口号。
socket.send(new DatagramPacket(request.getBytes(),request.getBytes().length, InetAddress.getByName(serverIp),port));
  1. 接收服务器的响应
DatagramPacket responsePacket = new DatagramPacket(new byte[2024],2024);
socket.receive(responsePacket);
  1. 从服务器读取的数据进行解析, 打印出来.
String response = new String(responsePacket.getData(),0,responsePacket.getData().length);System.out.println(response);
  1. 总代码
import java.io.IOException;
import java.net.*;
import java.util.Scanner;public class UdpEchoClient {//先创建一个成员变量:Socket对象,用于接收发送数据报private DatagramSocket socket = null;// UDP 本身不保存对端的信息, 自己的代码中保存一下private String serverIp;private int port;public UdpEchoClient(String serverIp,int port) throws SocketException {this.serverIp = serverIp;this.port = port;socket = new DatagramSocket();}public void start() throws IOException {while (true) {// 1. 从控制台读取用户输入的内容.System.out.println("请输入要传输的内容");Scanner scanner = new Scanner(System.in);if (!scanner.hasNext()) {break;}String request =scanner.next();//  发送数据报.把请求发送给服务器, 需要构造 DatagramPacket 对象。 构造过程中, 不光要构造载荷, 还要设置服务器的 IP 和端口号。socket.send(new DatagramPacket(request.getBytes(),request.getBytes().length, InetAddress.getByName(serverIp),port));//接收服务器的响应DatagramPacket responsePacket = new DatagramPacket(new byte[2024],2024);socket.receive(responsePacket);//从服务器读取的数据进行解析, 打印出来.String response = new String(responsePacket.getData(),0,responsePacket.getData().length);System.out.println(response);}}public static void main(String[] args) throws IOException {UdpEchoClient client = new UdpEchoClient("127.0.0.1",6666);client.start();}
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

web——sqliabs靶场——第十二关——(基于错误的双引号 POST 型字符型变形的注入)

判断注入类型 a OR 1 1# 发现没有报错 ,说明单引号不是闭合类型 测试别的注入条件 a) OR 1 1# a)) OR 1 1# a" OR 11 发现可以用双引号闭合 发现是")闭合 之后的流程还是与11关一样 爆破显示位 先抓包 是post传参,用hackbar来传参 unam…

【Linux】开发工具make/Makefile、进度条小程序

Linux 1.make/Makefile1.什么是make和Makefile?2.stat命令3.Makefile单个文件的写法4.Makefile多个文件的写法 2.进度条1.回车\r、换行\n2.缓冲区3.进度条1.倒计时程序2.进度条程序 1.make/Makefile 1.什么是make和Makefile? 一个工程中的源文件不计其…

Ubuntu22.04配置强化学习环境及运行相关Demo

什么是强化学习 强化学习(Reinforcement Learning,简称 RL)是机器学习中的一个重要分支,属于一种基于试错机制的学习方法。它通过让智能体(Agent)与环境(Environment)进行交互&…

GitHub 开源项目 Puter :云端互联操作系统

每天面对着各种云盘和在线应用,我们常常会遇到这样的困扰。 文件分散在不同平台很难统一管理,付费订阅的软件越来越多,更不用说那些烦人的存储空间限制了。 最近在 GitHub 上发现的一个开源项目 Puter 彻底改变了我的在线办公方式。 让人惊…

深入解析小程序组件:view 和 scroll-view 的基本用法

深入解析小程序组件:view 和 scroll-view 的基本用法 引言 在微信小程序的开发中,组件是构建用户界面的基本单元。两个常用的组件是 view 和 scroll-view。这两个组件不仅功能强大,而且使用灵活,是开发者实现复杂布局和交互的基础。本文将深入探讨这两个组件的基本用法,…

河道水位流量一体化自动监测系统:航运安全的护航使者

在广袤的水域世界中,航运安全始终是至关重要的课题。而河道水位流量一体化自动监测系统的出现,如同一位强大的护航使者,为航运事业的稳定发展提供了坚实的保障。 水位传感器:负责实时监测河道的水位变化。这些传感器通常采用先进的…

开源可视化工具对比:JimuReport VS DataEase

在当今数据驱动的时代,高效的数据可视化工具成为企业洞察业务、做出决策的关键利器。那对于企业来讲如何选择BI产品呢? 在开源可视化工具的领域中,JimuReport和DataEase 以其独特的优势脱颖而出,究竟谁更胜一筹呢?让我…

jquery还有其应用场景,智慧慢慢地被边缘化,但不会消亡

一、jQuery 的辉煌过往 jQuery 的诞生与崛起 在前端开发的漫长历史中,2006 年诞生的 jQuery 犹如一颗耀眼的新星划破天际。它由 John Resig 创造,一出现便以其独特的魅力迅速吸引了广大开发者的目光。在那个前端技术发展相对缓慢的时期,jQue…

TSmaster 专栏索引

文章目录 软件下载官方中文手册和视频教程窗口对齐关闭窗体报文格式转换TSmaster 硬件配置及连接TSmaster Measurement setup(测量设置) 软件下载 下载路径:https://www.tosunai.com/downloads/ 官方中文手册和视频教程 窗口对齐 一个工作…

Java小白成长记(创作笔记一)

目录 序言 思维导图 开发流程 新建SpringBoot并整合MybatisPlus 新建SpringBoot 整合MybatisPlus 统一结果封装 全局异常处理 引入数据库 序言 在一个充满阳光的早晨,一位对编程世界充满好奇的年轻人小小白,怀揣着梦想与激情,踏上了学习…

SpringBoot+Vue 2 多方法实现(图片/视频/报表)文件上传下载,示例超详细 !

目录 一、主流方法介绍 1. Base 64 2. 二进制流传输 3. multipart/form-data 4. FTP/SFTP 5. 云存储服务API 二、multipart/form-data 方式上传单个文件 1、前端部分 2、后端部分 三、multipart/form-data 方式上传多个文件 1、前端部分 2、后端部分 四、Base 64 方…

小米顾此失彼:汽车毛利大增,手机却跌至低谷

科技新知 原创作者丨依蔓 编辑丨蕨影 三年磨一剑的小米汽车毛利率大增,手机业务毛利率却出现下滑景象。 11月18日,小米集团发布 2024年第三季度财报,公司实现营收925.1亿元,同比增长30.5%,预估902.8亿元;…

unity中:超低入门级显卡、集显(功耗30W以下)运行unity URP管线输出的webgl程序有那些地方可以大幅优化帧率

删除Global Volume: 删除Global Volume是一项简单且高效的优化措施。实测表明,这一改动可以显著提升帧率,甚至能够将原本无法流畅运行的场景变得可用。 更改前的效果: 更改后的效果: 优化阴影和材质: …

webgl threejs 云渲染(服务器渲染、后端渲染)解决方案

云渲染和流式传输共享三维模型场景 1、本地无需高端GPU设备即可提供三维项目渲染 云渲染和云流化媒体都可以让3D模型共享变得简单便捷。配备强大GPU的远程服务器早就可以处理密集的处理工作,而专有应用程序,用户也可以从任何个人设备查看全保真模型并与…

React Native 基础

React 的核心概念 定义函数式组件 import组件 要定义一个Cat组件,第一步要使用 import 语句来引入React以及React Native的 Text 组件: import React from react; import { Text } from react-native; 定义函数作为组件 const CatApp = () => {}; 渲染Text组件

【专题】中国企业出海洞察报告暨解码全球制胜之道报告汇总PDF洞察(附原数据表)

原文链接:https://tecdat.cn/?p38314 在当今全球化的浪潮中,中国企业的出海行动正以前所未有的规模和速度展开,成为全球经济舞台上的重要力量。本报告旨在对 2024 年中国企业出海情况进行深度洞察,涵盖多个领域和视角。 从对外投…

『VUE』32. 动态组件,组件的动态切换(详细图文注释)

目录 动态组件示例代码总结 欢迎关注 『VUE』 专栏&#xff0c;持续更新中 欢迎关注 『VUE』 专栏&#xff0c;持续更新中 动态组件 有些时候我们的组件需要切换,这个时候用到<conponent></conponent>标签, <component :is"choseComponent"><…

传输层协议TCP

一.TCP协议格式 对于传输层协议我们之前是学过了UDP&#xff0c;对于传输层协议是存在了一定的了解的&#xff0c;所以现在我们再来看TCP协议格式&#xff1a; 我们之前学过UDP的报文格式&#xff0c;所以源端口和目的端口是不需要进行再次讲解的&#xff0c;对于32序号和确认序…

不能打开网页,但能打开QQ、微信(三种方式)

1.VPN错误 下面三个开关全关闭 2.DNS问题 WINR 输入CMD打开命令行 命令行输入 ipconfig/flushdns 重启电脑 3.直接火绒&#xff08;一键修复&#xff09;

Unity类银河战士恶魔城学习总结(P132 Merge skill tree with skill Manager 把技能树和冲刺技能相组合)

【Unity教程】从0编程制作类银河恶魔城游戏_哔哩哔哩_bilibili 教程源地址&#xff1a;https://www.udemy.com/course/2d-rpg-alexdev/ 本章节实现了解锁技能后才可以使用技能&#xff0c;先完成了冲刺技能的锁定解锁 Dash_Skill.cs using System.Collections; using System…