【TCP和UDP通信】多发多收

news/2024/5/18 15:59:55 标签: tcp/ip, udp, 网络

TCP和UDP通信——多发多收

UDP通信

1.客户端可以反复发送数据

客户端实现步骤

(1)创建DatagramSocket对象(发送端对象)

(2)使用while死循环不断的接收用户的数据输入,如果用户输入”886”则退出程序

(3)使用DatagramSocket对象的send方法将数据包对象进行发送

(4)释放资源

public class UDPClient {
    public static void main(String[] args) throws IOException {
        Scanner sc = new Scanner(System.in);
        DatagramSocket client = new DatagramSocket(); //使用随机端口

        while (true) {
            //发送的信息由键盘录入
            System.out.println("请输入聊天内容:");
            String info = sc.next();
            byte[] bytes = info.getBytes();
            DatagramPacket dp = new DatagramPacket(
                    bytes,
                    bytes.length,
                    InetAddress.getByName("127.0.0.1"),
                    20000
            );
            client.send(dp);

            //循环发送如果输入"886"退出
            if (info.contains("886")) {
                System.out.println("正在下线..");
                client.close();
                break;
            }
        }
    }
}
2.接收端可以反复接收数据
1.接受端可以反复接收数据

接收端实现步骤

(1)创建DatagramSocket对象并指定端口(接收端对象)

(2)创建DatagramSocket对象接收数据(数据包对象)

(3)使用DatagramSocket对象的receive方法传入DatagramPacket对象

(4)暂且不释放资源,使用while死循环不断的进行第3步

public class UDPServer {
    public static void main(String[] args) throws IOException {
        System.out.println("=== 聊天室 ===");
        DatagramSocket server = new DatagramSocket(20000);
        //循环接收
        while (true) {
            byte[] bytes = new byte[1024];
            DatagramPacket dp = new DatagramPacket(bytes, bytes.length);
            server.receive(dp);
            //展示数据
            System.out.println(new String(bytes,0, dp.getLength()));
            //DatagramPacket对象获取客户端其他信息
            System.out.println(dp.getAddress().getHostAddress() + "/" + dp.getPort());
            //不要关闭服务器Socket对象,否则只能接一次
            System.out.println("-------------------");
        }
    }
}

TCP通信

1.客户端可以反复发送数据

实现步骤:

(1)客户端使用死循环,让用户不断输入消息

(2)服务端也使用死循环,控制服务端接收完消息,继续等到接收下一个消息

public class TCPClient {
    public static void main(String[] args) throws Exception {
        Scanner sc = new Scanner(System.in);
        Socket client = new Socket("127.0.0.1", 8888);
        OutputStream os = client.getOutputStream();
        DataOutputStream dos = new DataOutputStream(os);

        while (true) {
            System.out.println("请输入联调内容:");
            String info = sc.next();
            dos.writeUTF(info);
            dos.flush();
            //循环发送如果输入"886"退出
            if (info.contains("886")) {
                System.out.println("正在下线..");
                client.close();
                dos.close();
                break;
            }
        }
    }
}
2.接收端可以反复接收数据
public class TCPServer {
    public static void main(String[] args) throws Exception {
        System.out.println("=== 聊天室 ===");
        ServerSocket server = new ServerSocket(8888);
        Socket serverSocket = server.accept();
        InputStream is = serverSocket.getInputStream();
        DataInputStream dis = new DataInputStream(is);
        //循环接收
        while (true) {
            try {
                String info = dis.readUTF(); //有用户下线,会抛异常SocketException
                System.out.println(info);
            } catch (Exception e) {
                //有人下线则释放资源
                System.out.println(serverSocket.getRemoteSocketAddress() + "离线了");
                System.out.println("-----------------------");
                server.close();
                dis.close();
                break;
            }
        }

    }
}

TCP通信-支持与多个客户端同时通信(多线程)

如何实现服务端同时接收多个客户端的消息的?

  • 主线程定义了循环负责接收客户端Socket管道连接
  • 没接收到一个Socket通信管道后分配给一个独立的线程负责处理它

客户端:

public class TCPClient {
    public static void main(String[] args) throws Exception {
        Scanner sc = new Scanner(System.in);
        Socket client = new Socket("127.0.0.1", 8888);
        OutputStream os = client.getOutputStream();
        DataOutputStream dos = new DataOutputStream(os);
        while (true) {
            System.out.println("请输入联调内容:");
            String info = sc.next();
            dos.writeUTF(info);
            dos.flush();
            //循环发送如果输入"886"退出
            if (info.contains("886")) {
                System.out.println("正在下线..");
                client.close();
                dos.close();
                break;
            }
        }
    }
}

任务类:封装本次连接的客户端要完成的逻辑

public class ServerThread implements Runnable {
    //serverSocket:本次连接的客户端的Socket对象
    private Socket serverSocket;
    public ServerThread(Socket serverSocket) {
        this.serverSocket = serverSocket;
    }
    @Override
    public void run() {
        try {
            //接收数据
            InputStream is = serverSocket.getInputStream();
            DataInputStream dis = new DataInputStream(is);
            //循环接收
            while (true) {
                try {
                    String info = dis.readUTF(); //有用户下线,会抛异常SocketException
                    System.out.println(info);
                } catch (Exception e) {
                    //有人下线则释放资源
                    System.out.println(serverSocket.getRemoteSocketAddress() + "离线了");
                    System.out.println("-----------------------");
                    serverSocket.close();
                    dis.close();
                    break;
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

服务端:使用线程循环接收

package com.demo12_TCP多发多收_多线程实现;

// TODO: 能够完成TCP通信下服务端可以同时接收多个客户端的消息

import java.net.ServerSocket;
import java.net.Socket;

public class TCPServer {
    public static void main(String[] args) throws Exception {
        System.out.println("=== 聊天室 ===");
        ServerSocket server = new ServerSocket(8888);

        while (true) {
            Socket serverSocket = server.accept();
            //每当服务器监听到一个连接请求,获取serverSocket对象,开启一个线程处理任务
            new Thread(new ServerThread(serverSocket)).start();
        }
    }
}

http://www.niftyadmin.cn/n/5064694.html

相关文章

线程概念,实现方式以及多线程模型

1.线程引入 有的进程可能需要“同时”做很多事,而传统的进程只能串行地执行一系列程序。 为此,引入了“线程”,来增加并发度。 可以把线程理解为“轻量级进程”。线程是一个基本的CPU执行单元,也是程序执行流的最小单位。引入线…

MacOS怎么配置JDK环境变量

1 输入命令看是否配置了JDk 的环境变量:echo $JAVA_HOME 要是什么也没输出 证明是没配置 2 输入命令编辑 sudo vim ~/.bash_profile 然后按 i ,进入编辑模式,粘贴下面的代码,注意:JAVA_HOME后面路径需要改成自己的版…

如何限制文件只能通过USB打印机打印,限制打印次数和时限并且无法在打印前查看或编辑内容

在今天这个高度信息化的时代,文档打印已经成为日常工作中不可或缺的一部分。然而,这也带来了诸多安全风险,如文档被篡改、知识产权被侵犯以及信息泄露等。为了解决这些问题,只印应运而生。作为一款独特的软件工具,只印…

从0手写两轮差速机器人urdf模型

文章目录 前言一、基本理论二、实现步骤1.创建一个机器人建模功能包2.使用圆柱体创建一个车体模型2.同理创建机器人其它构件3.机器人模型添加传感器 前言 最近为找到与自己课题应用场景相适应的机器人结构,对机器人建模方面的内容进行了了解和学习,计划…

CCF CSP认证 历年题目自练 Day20

题目一 试题编号: 201903-1 试题名称: 小中大 时间限制: 1.0s 内存限制: 512.0MB 问题描述: 题目分析(个人理解) 常规题目,先看输入,第一行输入n表示有多少数字&am…

DS线性表之链表

前言 我们上一期介绍了顺序表,它的底层就是数组,我们也分别对顺序表的动态版本和静态版本进行了实现!并且分析了顺序表的优缺点,优点是:尾插、尾删效率很高,其时间复杂度是O(1);缺点是&#xff…

操作系统-《王道 操作系统》

概念、功能和目标 概念 什么是操作系统 功能和目标 作为系统的管理者向上提供方便简易的服务作为最接近底层硬件的层次 特征 并发 共享 并发与共享的关系—互为存在条件 虚拟 异步 发展与分类 单道批处理系统多道批处理系统 优点:多道程序并发执行&#xff0c…

Autosar诊断实战系列19-UDS单帧数据接收代码逻辑分析

本文框架 前言1 CanIf→CanTp诊断数据的接收2. CanTp→Dcm单帧数据处理前言 在本系列笔者将结合工作中对诊断实战部分的应用经验进一步介绍常用UDS服务的进一步探讨及开发中注意事项, Dem/Dcm/CanTp/Fim模块配置开发及注意事项,诊断与BswM/NvM关联模块的应用开发及诊断capl测…