[网络编程]TCP和UDP的比较 及 通过java用UDP实现网络编程

2024-09-07 08:12

本文主要是介绍[网络编程]TCP和UDP的比较 及 通过java用UDP实现网络编程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一. 网络编程套接字
  • TCP和UDP的特点
    • 有连接 vs 无连接
    • 可靠传输 vs 不可靠传输
    • 面向字节流 vs 面向数据报
    • 全双工 vs 半双工
  • 二. java用UDP实现网络编程
    • 代码实现:
    • 运行代码:
    • 启动多个客户端
    • 别人能否使用?
    • 实现翻译功能

一. 网络编程套接字

网络编程套接字, 就是指操作系统提供的网络编程的api, 称为"socket api"

操作系统, 提供的socket api, 有三套:

  1. 流式 套接字 => 给TCP使用的
  2. 数据报 套接字 => 给UDP使用的
  3. Unix域套接字 =>不能夸主机通信, 只能本地主机上的进程和进程之间的通信(现在很少使用)

TCP和UDP都是传输层协议, 都是给应用程序提供服务的
但是这两个协议特点差异非常大, 因此我们就需要搞两套api, 来分别表示

TCP和UDP的特点

TCP的特点:
有连接, 可靠传输, 面相字节流, 全双工
UDP的特点:
无连接, 不可靠传输, 面向数据报, 全双工

有连接 vs 无连接

计算机中的连接, 指的是建立联系的双方, 各自保存对方的信息
那么TCP协议要求:
通信双方保存对方的信息
UDP要求:
通信双方不需要保存对方的信息

可靠传输 vs 不可靠传输

可靠 != 安全
可靠, 指的是要传输的数据, 尽可能的传输给对方, 尽力不是确保
那么, TCP协议, 内部提供了一系列的机制, 来实现可靠传输
UDP协议, 只关心发送, 并不关心后续

面向字节流 vs 面向数据报

文件操作, 也是字节流的, 读写非常灵活, 读写的基本单位就是字节
TCP也是字节流的

数据报Datagram, 数据段Segment, 数据包Packet, 数据帧Frame, 这些都是网络传输数据的基本单位, 但是有不同的使用场景
那么, UDP, 是面向数据报, 传输数据的基本单位, 是一个个的UDP数据报

全双工 vs 半双工

全双工: 一条链路, 能够进行双向通信
半双工: 一条链路, 只能进行单向通信
TCP, UDP都是全双工

后续会对TCP和UDP进行进一步详细介绍

二. java用UDP实现网络编程

socket api 都是系统提供的, java中对于系统的这些api进行进一步封装了
UDP的socket api 重点的两个类:
1. DatagramSocket
系统中的socket, 可以理解成是一种"文件", socket文件, 可以视为是"网卡"这种硬件设备的抽象表现形式
DatagramSocket, 就是对socket进行封装, 可以视为是"操作网卡"的遥控器
针对这个对象进行读写操作, 就是在针对网卡进行读写操作

具有"遥控器属性"这样的概念, 计算机中起了个专门的名字"句柄"(handle)

构造方法:
在这里插入图片描述
读写方法:
在这里插入图片描述

2. DatagramPacket
是针对UDP数据报的一个抽象表示

构造方法:
在这里插入图片描述
成员方法:
在这里插入图片描述

代码实现:

网络程序, 既有服务器, 也有客户端
我们现在只是进行一个简单的实现, 服务器收到客户端请求后, 无需做什么, 客户端请求啥, 直接响应啥, 这种程序, 就叫做"回显"(Echo)
服务器:
第一步:创建对象
在这里插入图片描述

  1. 服务器运行, 要把端口号确定下来, 以便让客户端找到, 客户端才能主动发送请求
    实际运行的时候, 端口只能为多少是程序猿自己决定的
    但是要确保: 1)端口号是合法的 1-65535 2) 自己写的端口号, 不能和别的进程使用的端口号冲突
  2. 为什么把new对象放在构造方法里? 方便抛异常

第二步:创建start方法
在这里插入图片描述
服务器需要不停的处理请求, 不停地响应, 所以需要写成死循环

第三步:读取请求并解析
在这里插入图片描述
这个对象是一个UDP数据报, 包含两个部分:
1)报头(通过类的属性来表示的)
2)载荷(通过构造方法传递的字节数组, 作为持有载荷的空间, 这个存储载荷的空间, 人家没有自己new, 而是让你new好)

在这里插入图片描述
如果receive没有收到请求, 那么就会阻塞等待, 直到有客户端发送请求
为了方便后续再java代码中的处理(打印), 可以把上述数据报中的二进制数据拿出来, 构造成String

第四步:根据请求计算响应
在这里插入图片描述
将响应的结果通过process方法返回:
在这里插入图片描述

第五步: 把响应写回到客户端
在这里插入图片描述
构造一个DatagramPacket作为发送的数据报
第一个参数是byte[] 类型, String中的getBytes()方法可以获取字节
最后一个参数是用来获取客户端的ip和端口信息

前面我们知道, UDP是无连接的, 所以本身并没有存储对方的ip和端口信息, 但是上述我们用来接收数据的DatagramPacket中, 存放了客户端的ip和接口, 所以我们可以通过getSocketAddress()方法来获取到

第六步: 打印请求记录
在这里插入图片描述
这样我们就完成了UDP协议下的服务器实现!

完整代码:

public class UDPEchoServer {private DatagramSocket socket = null;public UDPEchoServer(int port) throws SocketException {socket = new DatagramSocket(port);}public void start() throws IOException {System.out.println("服务器启动!");while(true){//1)读取请求并解析DatagramPacket requestPacket = new DatagramPacket(new byte[4096], 4096);socket.receive(requestPacket);String request = new String(requestPacket.getData(), 0, requestPacket.getLength());//2)根据请求计算响应String response = this.process(request);//3)把响应写回到客户端DatagramPacket responsePacket = new DatagramPacket(response.getBytes(), 0, response.getBytes().length,requestPacket.getSocketAddress());socket.send(responsePacket);System.out.printf("[%s:%d] req=%s, resp=%s\n", requestPacket.getAddress(), requestPacket.getPort(),request, response);}}private String process(String request) {return request;}public static void main(String[] args) throws IOException {UDPEchoServer server = new UDPEchoServer(9090);//随便假设的端口号, 如果不能运行就换一个server.start();}}

客户端:
第一步:创建对象
因为客户端是先发送请求的一端, 所以当我们创建类的时候, 需要知道服务器的ip地址和端口
在这里插入图片描述
注意: 客户端这边, 创建DatagramSocket对象时, 不需要传端口参数
不是客户端不需要端口, 是因为操作系统, 自动分配了一个空闲的(不和别人冲突)的端口号, 不需要自己指定
这个自动分配的端口号, 每次重新启动程序都可能不一样

第二步: 创建start方法
在这里插入图片描述
客户端可以不停地给服务器发送请求, 所以使用死循环

第三步: 从控制台读取用户输入
在这里插入图片描述

第四步: 构造请求并发送
在这里插入图片描述
因为客户端和服务器之间是没有连接的, 所以我们怎么才能知道将数据传给谁呢
数据报帮我们完成了这个工作, 在构造数据报时, 我们需要传5个参数,
前三个和写服务器是一样, 将字符串转化成字节数组, 并规定长度
后两个是传服务器的IP地址和端口号
传服务器的IP地址时, 我们所获取到的是点分十进制的字符串, 我们需要转化成java能够识别的对象
需要借助InetAddress.getByName方法

第五步: 读取响应的数据
在这里插入图片描述
需要自己new空间存放数据

第六步:显示响应到控制台
在这里插入图片描述
将接收到的数据报转成字符串, 打印到控制台上

第七步: 实现main方法
在这里插入图片描述
当前服务器和客户端在一个主机上, 固定使用"127.0.0.1" 这个ip即可, 称为环回ip

完整代码:

public class UdpEchoClient {private DatagramSocket socket = null;private String serverIp;private int serverPort;public UdpEchoClient(String serverIp, int serverPort) throws SocketException {socket = new DatagramSocket();this.serverIp = serverIp;this.serverPort = serverPort;}public void start() throws IOException {System.out.println("客户端启动!");Scanner scanner = new Scanner(System.in);while(true){System.out.println("请输入要发送的请求");//1. 从控制台读取用户输入String request = scanner.next();//2. 构造请求并发送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();}
}

运行代码:

要先运行服务器, 再运行客户端
在这里插入图片描述
在这里插入图片描述
这样, 客户端和服务器就能够正常运行啦

启动多个客户端

默认情况下, idea里面只能启动一个客户端, 要想启动多个:
在这里插入图片描述
在这里插入图片描述
这样就可以运行多个客户端啦
在这里插入图片描述
在这里插入图片描述

别人能否使用?

我们这个客户端和服务器都在同一个主机上, 那么不同如果是不同主机, 是不能成功的
因为:
我们的电脑的IP只是一个局域网内部使用的私有IP, 而不是在广域网上直接使用的"公有IP"
虽然我们没有公有IP, 但是我们可以买云服务器

实现翻译功能

在这里插入图片描述
我们上述逻辑, 属于回显, 并没有实际完成什么逻辑
接下来我们来实现一个简单的翻译功能的服务器
由于逻辑大致相同, 只有process内部的功能不同, 所以我们可以使用继承来完成

 public class UdpDictServer extends UDPEchoServer{private Map<String, String> dict = null;public UdpDictServer(int port) throws SocketException {super(port);dict = new HashMap<>();dict.put("hello", "你好");dict.put("cat", "小猫");dict.put("dog", "小狗");}@Overridepublic String process(String request) {return dict.getOrDefault(request, "没查询到该词汇");}public static void main(String[] args) throws IOException {UDPEchoServer server = new UdpDictServer(9090);server.start();}
}

在这里插入图片描述
在这里插入图片描述

这篇关于[网络编程]TCP和UDP的比较 及 通过java用UDP实现网络编程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1144540

相关文章

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.