【Java网络编程】基于UDP-Socket 实现客户端、服务器通信

简介: 【Java网络编程】基于UDP-Socket 实现客户端、服务器通信

 

Socket 套接字可以理解为是操作系统提供给程序员的一组用于网络编程的API (接口)——传输层的接口,传输层给应用层提供的一组 API,统称为 Socket API 。网络通信的底层逻辑都已经被操作系统封装好了,开发人员就可以根据接口开发实现网络通信。

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

    • 字节流套接字:使用传输层TCP协议
    • 数据报套接字:使用传输层UDP协议
    • 原始套接字:用于自定义传输层协议

    本篇博客主要讲述面向数据报的网络编程。


    一、Java基于UDP数据报套接字通信模型

    UDP 协议的特点:

      1. 面向无连接:传输数据之前,通信双方不依赖于建立连接,只需要知道谁发给谁即可。
      2. 不可靠传输:只负责发送,不关注数据是否传输成功,即使没发送成功啥也不干,也没有反馈
      3. 面向数据报:使用 UDP 数据报的形式传输,数据报可以理解为数据是一块一块的传输。image.gif编辑
      4. 有缓存区的概念:接收缓冲区,发送缓冲区:在进行网络数据通信时,用于存储待发送或已接收数据的缓冲区。
      5. 数据传输大小受限制:一个UDP 数据报最大占 64k, 1k = 1024字节
      6. 全双工通信:通信双方都可以同时进行信息交互

      总结:UDP 协议,具有面向无连接,面向数据报特点,即使通信双方没有建立连接,也会传输数据,并且一次性发送全部的数据报,一次性接受全部的数据报。

      Java 中基于UDP协议通信,主要使用 DatagramSocket 类来创建数据报套接字,并使用
      DatagramPacket 作为发送或接收的UDP数据报。


      针对一个客户端对服务器发出一次请求,服务器针对该请求给予响应流程如下:

      image.gif编辑在真实的网络环境中,一个服务器往往会给多个客户端提供响应。

      举个实际的例子根据以上流程图理解一下,张三老铁使用手机百度浏览器搜索 “美女图片”,此时前端页面就会将这条搜索记录 “美女图片” 拿到后台打包成UDP 数据报,以请求的形式发给百度服务器,百度服务器拿到请求后,解析数据报查看:哦~ 要美女啊,我这有很多,于是将“本地”存储的美女图片作为 UDP 数据报的形式响应给请求端,百度浏览器拿到百度服务器响应的数据后,通过解析,将图片展示在张三的手机界面上。


      二、UDP 数据报套接字编程

      2.1  DatagramSocket API

      DatagramSocket 是Socket 套接字基于UDP 协议来发送和接受数据报的类.

      Datagram : 数据报

      Socket : 说明这个对象是一个 Socket 对象

      Socket 对象相当于操作系统一个特殊的文件,这个文件并非对应对应的硬盘上的某个数据存储区域,而是对应到,网卡这个硬件!!!

      往 socket 对象中写数据,相当于通过网卡发送消息

      从 socket 对象中读数据,相当于通过网卡接收消息

      所以想要进行网络编程使得两个设备相互通信,就需要有 socket 文件这样的对象,借助这个 socket 文件对象来间接的操作网卡,操作系统基本设计思想:一切皆文件,为了简化系统内核的逻辑设计,socket 对象面向字节流读写


      image.gif编辑

      此处Socket 对象可以被客户端 / 服务端使用,服务器这边的 Socket 必须要关联一个具体的端口号,每个联网的设备在启动时或随机或指定都会绑定一个端口号,作为该应用程序的唯一标识,就相当于数据被你的设备接收,经解析需要知道这些数据需要交给那个应用程序处理。例如,qq 、微信没见过消息发串了的情况吧。

      客户端这边可以不需要手动指定,系统会自动随机分配,端口号的取值范围是 [0, 65535]。

      小于等于1024 的端口号,会提供给知名的服务器使用,不建议使用这一类。


      image.gif编辑

      socket 也是文件,用完了要关闭资源。


      2.2  DatagramPacket API

      DatagramPacket 类是用于基于UDP 协议的Socket 发送和接收的数据报。

      image.gif编辑

      只有两个参数的版本,不需要设计地址,通常用来接收消息。另一个多参数的版本需要显式的设置地址进去,通常要用来发送消息。


      image.gif编辑

      构造UDP 发送的数据报时,需要传入 socketAddress对象,这个对象可以使用 InetSocketAddress 来创建 。

      InetSocketAddress API 的构造方法:

      image.gif编辑


      三、基于UDP Socket 实现客户端,服务器程序

      分析客户端的程序的功能:

      1.发出请求(消息)

      2. 等待服务器响应(回应)

      3. 解析服务器的响应

      分析服务端的程序的功能:

      1. 尝试读取客户端的请求并解析

      2. 根据客户端的请求,统筹响应的数据

      3. 把响应返回到客户端


      3.1 服务端程序设计

      首先我们定义一个服务端的类:UdpEchoServer

      定义一个 Socket 对象文件,与网卡交互。

      //需要线定义一个 socket 对象
      //通过网络通信,必须要使用 socket 对象
      private DatagramSocket socket = null;

      image.gif

      利用 UdpEchoServer类 的构造方法为 socket 构造,并指定需要绑定的端口号

      public UdpEchoServer(int serverPort) throws SocketException {
              //构造 socket 的同时,指定要的关联/ 绑定的端口
              socket = new DatagramSocket(serverPort);
        }

      image.gif

      绑定一个端口,不一定能成功,如果某个端口已经被别的进程占用了,此时这里的绑定操作就会出错,同时一个主机上,一个端口,同一时刻,只能被一个进程绑定。所以这里会抛出异常。


      定义一个启动方法:start()

      方法内部采用循环的方式,每次循环,做三件事:

      1. 读取请求并解析,2. 根据请求计算响应,3. 把响应的结果写回客服端

      public void start() throws IOException {
              System.out.println("服务器启动!");
              while (true) {
                  // 每次循环,要做三件事情:
                  // 1. 读取请求并解析
                  // 构造接受请求的“空盒子”,指定盒子的大小 4096 字节
                  DatagramPacket requestPacket = new DatagramPacket(new byte[4096], 4096);
                  // 数据从网卡上来的
                  socket.receive(requestPacket);// 获取客户端的请求、
                  //为了方便处理这个请求,把二进制数据转换成 String
                  String request = new String(requestPacket.getData(), 0, requestPacket.getLength());
                  // 2. 根据请求计算响应 调用方法
                  String response = process(request);
                  /* 3. 把响应结果写回客服端
                  根据response 字符串,构造一个 DatagramPacket
                  和请求 packet 不同,此处构造响应的时候,需要指定这个包要发给谁 */
                  DatagramPacket responsePacket = new DatagramPacket(response.getBytes(), response.getBytes().length, requestPacket.getSocketAddress());
                  // response.getBytes() 转换为 byte二进制流
                  // requestPacket 是从客户端这里收来的,getSocketAddress 就会得到客户端的 IP 和 端口
                  socket.send(responsePacket);// 响应
                  // 可以保存以下日志
                  String now3 = df.format(System.currentTimeMillis()); // 返回当前系统时间
                  try(Writer writer = new FileWriter("outPut.txt", true)) {
                      writer.write(requestPacket.getAddress().toString() +
                              requestPacket.getPort() + " " + now3 +
                              " req: " + request + " resp:  " + response + "\n");
                  }
                  // 打印一下本次客户端与服务端的交互
                  System.out.printf("[%s:%d] req: %s, resp: %s\n", requestPacket.getAddress().toString(),
                          requestPacket.getPort(), request, response);
              }
          }

      image.gif

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

      image.gif

      2. 根据客户端的请求,统筹响应的数据,我们将第二步计算响应以函数的方式封装实现,参数 request 请求我们已经转换为字符串了,所以我们就可以依据请求做出对应的处理了,如果是需要搜索什么,就可以根据请求的内容将服务器的中存储的被搜索内容作为响应返回给客户端,如果是互相发消息,那就需要根据请求判断需要发给谁,此时需要双方约定在请求中添加接收方的IP 和 端口,这样才能找到接收方,作为服务器端来讲就需要使用多线程的思想,每一个客户端发来请求都会分配一个线程来处理。

      public String process(String request) { //这里直接将客户端的请求,作为响应返回
              return "服务器响应:" + request;
          }

      image.gif

      这个部分可以依据客户端的请求,进行对应的业务处理。


      3.2 客户端程序设计

      首先定义一个客户端的类:UdpEchoClient

      定义一个 Socket 对象文件,与网卡交互。

      定义 serverIP 的成员变量,用于存储服务器端的IP 地址

      定义 serverPort 的成员变量,用于存储服务器的端口号

      private DatagramSocket socket = null;
      private String serverIP = null; // IP
      private int serverPort = 0; // 端口号 —— 代表进程的标识

      image.gif

      利用 UdpEchoClient 类 的构造方法为 socket 构造,并为成员方法关联服务器的 IP 地址和端口

      //客户端启动,需要知道服务器在哪儿!!
          public UdpEchoClient(String serverIP, int serverPort) throws SocketException {
              //对于客户端来说,不需要显示关联端口
              // 不代表没有端口,而是系统自动分配了个空闲的端口
              socket = new DatagramSocket(); //获取随机端口号
              this.serverIP = serverIP;
              this.serverPort = serverPort;
          }

      image.gif

      定义一个启动方法:start()

      方法内部采用循环的方式,每次循环,做4件事:

      1. 从控制台获取用户输入的信息

      2. 将获取的信息构造成 UDP 数据报,并向服务器发送请求

      3. 客户端尝试获取读取服务器返回的响应

      4. 将获取的的响应根据业务需求做进一步的处理

      public void start() throws IOException {
              // 通过这个客户端可以多次和服务器进行交互
              System.out.println("客户端启动!");
              Scanner in = new Scanner(System.in);
              while (true) {
                  //1.先从控制台,读取一个字符串过来
                  //先打印一个提示符,提示用户要输入内容
                  System.out.print("->");
                  String request = in.next();
                  //2. 把字符串构造成 UDP packet, 并进行发送
                  DatagramPacket requestPacket = new DatagramPacket(
                          request.getBytes(),
                          request.getBytes().length,
                          //----如果参数为null,获得的是本机的IP地址
                          InetAddress.getByName(serverIP),// 确定主机的 IP 地址----在给定主机名的情况下确定主机的IP地址
                          serverPort);
                  socket.send(requestPacket); //响应
                  //3. 客户端尝试读取服务器返回的响应
                  DatagramPacket responsePacket = new DatagramPacket(new byte[4096], 4096);
                  socket.receive(responsePacket);
                  //4. 把响应的数据转换成 String 显示出来,利用 String 的构造方法转换
                  String response = new String(responsePacket.getData(), 0, responsePacket.getLength());
                  System.out.printf("req: %s, resp: %s\n", request, response);
              }
          }

      image.gif

      public static void main(String[] args) throws IOException {
              UdpEchoClient udpEchoClient = new UdpEchoClient("127.0.0.1",9090);
              udpEchoClient.start(); // 启动客户端
          }

      image.gif


      3.3 总结:

      以上客户端服务器代码使用了 Socket “ 文件流”, 但是没有调用 close() 方法释放资源,是因为文件流会随着进程的关闭而关闭,但是在实际开发中一定要记得随手关闭哦~

      启动客户端和服务器展示效果:

      image.gif编辑

      image.gif编辑

      image.gif编辑

      以上操作流程执行图。image.gif编辑

      一定是先启动服务器,再启动客户端,

      UDP(User Datagram Protocol)是一个无连接的传输层协议,它不保证数据包可靠传输。UDP比TCP更快,而且在对网络延迟要求较低、对数据准确性要求不高的情况下使用。本文将讨论基于UDP Socket的客户端服务器开发总结。

      UDP Socket

      Socket是一种用于在计算机之间进行通信的 API(应用程序接口)。UDP Socket是基于UDP协议进行通信的套接字,它提供了一种快速发送和接收数据包的方法,但是与TCP不同,它不会重复数据或者检查是否有缺失的数据包。

      编写UDP客户端的基本步骤如下:

        1. 创建一个UDP socket。
        2. 将要发送的数据打包到UDP数据包中。
        3. 使用UDP socket发送数据包到特定的IP地址和端口号。

        编写UDP服务器的基本步骤如下:

          1. 创建一个UDP socket并监听特定的端口号。
          2. 接收客户端发送的数据包。
          3. 处理数据包,然后将响应打包到另一个UDP数据包中。
          4. 使用UDP socket向客户端发送响应数据包。

          总结:

          在使用UDP Socket进行客户端和服务器开发时,需要注意以下几点:

            • UDP不会保证数据包的可靠传输,因此需要在应用程序中处理数据包的丢失或者损坏情况。
            • 由于UDP没有连接状态,因此可以同时与多个客户端通信。
            • 在UDP通信中,需要指定目标地址和端口号。
            相关文章
            |
            3月前
            |
            JSON 网络协议 安全
            【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
            几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
            223 1
            |
            3月前
            |
            JSON 网络协议 安全
            【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
            几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
            241 1
            |
            10月前
            |
            Java Linux 定位技术
            Minecraft配置文件参数说明(JAVA服务器篇)
            Minecraft JAVA版服务器启动后会生成server.properties配置文件,位于minecraft_server/根目录下。该文件包含多项关键设置,如游戏模式(gamemode)、最大玩家数(max-players)、难度(difficulty)等。此文档详细说明了各配置项的功能与默认值,帮助用户高效管理服务器环境。
            2300 60
            |
            10月前
            |
            网络协议 开发者 Python
            Socket如何实现客户端和服务器间的通信
            通过上述示例,展示了如何使用Python的Socket模块实现基本的客户端和服务器间的通信。Socket提供了一种简单且强大的方式来建立和管理网络连接,适用于各种网络编程应用。理解和掌握Socket编程,可以帮助开发者构建高效、稳定的网络应用程序。
            539 10
            |
            10月前
            |
            前端开发 Cloud Native Java
            Java||Springboot读取本地目录的文件和文件结构,读取服务器文档目录数据供前端渲染的API实现
            博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
            Java||Springboot读取本地目录的文件和文件结构,读取服务器文档目录数据供前端渲染的API实现
            |
            10月前
            |
            存储 网络协议 安全
            Java网络编程,多线程,IO流综合小项目一一ChatBoxes
            **项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
            452 23
            |
            11月前
            |
            网络协议 算法 安全
            Go语言的网络编程与TCP_UDP
            Go语言由Google开发,旨在简单、高效和可扩展。本文深入探讨Go语言的网络编程,涵盖TCP/UDP的基本概念、核心算法(如滑动窗口、流量控制等)、最佳实践及应用场景。通过代码示例展示了TCP和UDP的实现,并讨论了其在HTTP、DNS等协议中的应用。最后,总结了Go语言网络编程的未来发展趋势与挑战,推荐了相关工具和资源。
            347 5
            |
            存储 监控 安全
            单位网络监控软件:Java 技术驱动的高效网络监管体系构建
            在数字化办公时代,构建基于Java技术的单位网络监控软件至关重要。该软件能精准监管单位网络活动,保障信息安全,提升工作效率。通过网络流量监测、访问控制及连接状态监控等模块,实现高效网络监管,确保网络稳定、安全、高效运行。
            290 11
            |
            监控 网络协议 网络性能优化
            网络通信的核心选择:TCP与UDP协议深度解析
            在网络通信领域,TCP(传输控制协议)和UDP(用户数据报协议)是两种基础且截然不同的传输层协议。它们各自的特点和适用场景对于网络工程师和开发者来说至关重要。本文将深入探讨TCP和UDP的核心区别,并分析它们在实际应用中的选择依据。
            466 3
            |
            Java
            java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
            java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
            263 9