java案例-服务端与客户端(传输对象)

需求

在这里插入图片描述

代码

  • SysUser 用户类
  • Operation 操作类
  • Client 客户端
  • Server 服务端
  • ServerReaderThread 服务端线程类

SysUser 用户类

  • 需要实现Serializable 方便序列化,传输对象
public class SysUser implements Serializable {
    private String username;
    private String password;

    public SysUser() {
    }

    public SysUser(String username, String password) {
        this.username = username;
        this.password = password;
    }

    public String getUsername() {
        return this.username;
    }

    public String getPassword() {
        return this.password;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    @Override
    public String toString() {
        return "SysUser{" +
                "username='" + username + '\'' +
                ", password='" + password + '\'' +
                '}';
    }
}

操作类

操作方法和被操作的用户

public class Operation implements Serializable {
    private String operation;
    private SysUser sysUser;

    public Operation() {
    }

    public Operation(String operation, SysUser sysUser) {
        this.operation = operation;
        this.sysUser = sysUser;
    }

    public String getOperation() {
        return this.operation;
    }

    public SysUser getSysUser() {
        return this.sysUser;
    }

    public void setOperation(String operation) {
        this.operation = operation;
    }

    public void setSysUser(SysUser sysUser) {
        this.sysUser = sysUser;
    }

    @Override
    public String toString() {
        return "Operation{" +
                "operation='" + operation + '\'' +
                ", sysUser=" + sysUser +
                '}';
    }
}

客户端

  • 本例是一个长连接:一直挂在服务器端,那么服务器端的读取线程中就必须要有一个while 循环 不断读取此长连接的内容
  • 如果不使用本例的方法,而是每次都创建新的socket 连接服务器,那么读取线程中就必须要有一个while 循环就是非必要的,因为每次连接只会发送一次,等待服务器端响应之后,需要关闭此socket,这就是短连接
// 该类是一个客户端类
public class Client {

    private static Scanner scanner = new Scanner(System.in);



    public static void main(String[] args) throws IOException {
        start();
    }


    // 该方法是一个启动方法
    public static void start() throws IOException {

        Scanner scanner = new Scanner(System.in);

        Socket socket = new Socket("127.0.0.1",8888);

        // 此处加死循环 是为了让客户端一直运行
        while (true){

            System.out.println("请输入你要进行的操作");
            System.out.println("1.注册");
            System.out.println("2.登录");
            System.out.println("3.退出");
            String i = scanner.next();
            switch (i){
                case "1":
                    System.out.println("注册");
                    register(socket);
                    break;
                case "2":
                    System.out.println("登录");
                    login(socket);
                    break;
                case "3":
                    System.out.println("退出");
                    socket.close();
                    break;
                default:
                    System.out.println("输入错误,请重新输入");
            }
        }
    }

    // 该方法是一个注册方法
    public static void register(Socket socket) throws IOException {

        System.out.println("请输入用户名");
        String username = scanner.next();
        System.out.println("请输入密码");
        String password = scanner.next();
        SysUser sysUser = new SysUser(username,password);
        Operation operation = new Operation("register",sysUser);
        // 获取输出流
        requestAndRespond(socket, operation);

    }

    private static void requestAndRespond(Socket socket, Operation operation) throws IOException {
        OutputStream outputStream = socket.getOutputStream();
        // 封装输出流为对象输出流
        ObjectOutputStream objectOutputStream = new ObjectOutputStream(outputStream);
        // 写出对象
        objectOutputStream.writeObject(operation);
        objectOutputStream.flush();
        // 不要关闭流 否则会关闭socket
        //objectOutputStream.close();

        // 获取响应
        InputStream inputStream = socket.getInputStream();
        DataInputStream dataInputStream = new DataInputStream(inputStream);
        String response = dataInputStream.readUTF();
        System.out.println(response);

    }

    // 该方法是一个登录方法
    public static void login(Socket socket) throws IOException {
        System.out.println("请输入用户名");
        String username = scanner.next();
        System.out.println("请输入密码");
        String password = scanner.next();
        SysUser sysUser = new SysUser(username,password);
        Operation operation = new Operation("login",sysUser);
        // 获取输出流
        requestAndRespond(socket, operation);
    }
}

服务器端

// 该类是一个服务端类
public class Server {

    // 模拟数据库
    public static final File file = new File("test999.txt");

    public static ServerSocket serverSocket;

    //  创建线程池
    public static final ExecutorService pool =  new ThreadPoolExecutor(3, 5, 8, TimeUnit.SECONDS,
            new ArrayBlockingQueue<>(4), Executors.defaultThreadFactory(), new ThreadPoolExecutor.AbortPolicy());

    static {
        try {
            serverSocket = new ServerSocket(8888);
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }

    public static void main(String[] args) throws IOException {
        start();

    }

    // 该方法是一个启动方法
    public static void start() throws IOException {
        System.out.println("服务端启动");
        listen();
    }

    public static void listen() throws IOException {
        System.out.println("监听中");


        // 此处加死循环 是为了让服务器已知监听有没有客户端连接
        while (true){
            // 获取客户端的socket
            Socket socket = serverSocket.accept();
            // 使用线程池执行线程
            pool.execute(new ServerReaderThread(socket));
        }
    }

    // 该方法是一个注册方法
    public static void register(SysUser sysUser,Socket socket) throws IOException, ClassNotFoundException {
        System.out.println("注册");
        // 获取数据库中的所有用户
        FileInputStream fileInputStream = new FileInputStream(file);
        if (fileInputStream.available() > 0) {
            ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);
            ArrayList<SysUser> sysUsers = (ArrayList<SysUser>) objectInputStream.readObject();
            for (SysUser user : sysUsers) {
                if (user.getUsername().equals(sysUser.getUsername())) {
                    System.out.println("用户名已存在");
                    OutputStream outputStream = socket.getOutputStream();
                    DataOutputStream dataOutputStream = new DataOutputStream(outputStream);
                    dataOutputStream.writeUTF("用户名已存在");
                    dataOutputStream.flush();
//                    dataOutputStream.close();
                    return;
                }
            }
        }


        // 注册用户
        // 将用户写入数据库 追加模式
        FileOutputStream fileOutputStream = new FileOutputStream(file,true);
        ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);
        ArrayList<SysUser> sysUsersRegister = new ArrayList<>();
        sysUsersRegister.add(sysUser);
        objectOutputStream.writeObject(sysUsersRegister);
        System.out.println("注册成功");
        System.out.println(sysUser);
        OutputStream outputStream = socket.getOutputStream();
        DataOutputStream dataOutputStream = new DataOutputStream(outputStream);
        dataOutputStream.writeUTF("注册成功");
        dataOutputStream.flush();
//        dataOutputStream.close();


    }

    // 该方法是一个登录方法
    public static void login(SysUser sysUser,Socket socket) throws IOException, ClassNotFoundException {
        System.out.println("登录");
        // 获取数据库中的所有用户
        FileInputStream fileInputStream = new FileInputStream(file);
        if (fileInputStream.available() > 0) {
            ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);
            ArrayList<SysUser> sysUsers = (ArrayList<SysUser>) objectInputStream.readObject();
            for (SysUser user : sysUsers) {
                if (user.getUsername().equals(sysUser.getUsername())&&user.getPassword().equals(sysUser.getPassword())) {
                    System.out.println("登录成功");
                    System.out.println(sysUser);
                    OutputStream outputStream = socket.getOutputStream();
                    DataOutputStream dataOutputStream = new DataOutputStream(outputStream);
                    dataOutputStream.writeUTF("登录成功");
                    dataOutputStream.flush();
//                    dataOutputStream.close();
                }else {
                    System.out.println("用户名或密码错误");
                    OutputStream outputStream = socket.getOutputStream();
                    DataOutputStream dataOutputStream = new DataOutputStream(outputStream);
                    dataOutputStream.writeUTF("用户名或密码错误");
                    dataOutputStream.flush();
//                    dataOutputStream.close();
                }
            }
        }else {
            System.out.println("数据库中没有用户");
            OutputStream outputStream = socket.getOutputStream();
            DataOutputStream dataOutputStream = new DataOutputStream(outputStream);
            dataOutputStream.writeUTF("数据库中没有用户");
            dataOutputStream.flush();
//            dataOutputStream.close();
        }



    }
}

服务器端读取类

  • 本例是一个长连接:一直挂在服务器端,那么服务器端的读取线程中就必须要有一个while 循环 不断读取此长连接的内容
  • 如果不使用本例的方法,而是每次都创建新的socket 连接服务器,那么读取线程中就必须要有一个while 循环就是非必要的,因为每次连接只会发送一次,等待服务器端响应之后,需要关闭此socket,这就是短连接
ServerReaderThread extends Thread {
    private Socket socket;

    public ServerReaderThread(Socket socket) {
        this.socket = socket;
    }

    @Override
    public void run() {
        try {
            // 最开始这个地方没有死循环,导致客户端只能发送一次请求
            // 这个地方加入死循环 是为了即使是一个request也能一直获得用户的输入
            // 这个地方如果不加入死循环,那么应该建立短连接
            // 也就是说,客户端发送一个请求,都会建立一个新的socket,服务端处理完这个请求之后,返回给客户端结果,客户端处理完结果之后,关闭socket
            while(true){
                // 获取输入流
                InputStream inputStream = socket.getInputStream();
                // 封装输入流为对象输入流
                ObjectInputStream objectInputStream = new ObjectInputStream(inputStream);

                // 读取对象
                Operation operation = (Operation) objectInputStream.readObject();
                // 获取操作
                String operation1 = operation.getOperation();
                // 获取用户
                SysUser sysUser = operation.getSysUser();
                System.out.println("线程方法");
                switch (operation1) {
                    case "register":
                        register(sysUser,socket);
                        break;
                    case "login":
                        login(sysUser,socket);
                        break;
                    default:
                        break;
                }
            }

        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Swift - 枚举

文章目录 Swift - 枚举1. 枚举的基本用法2. 关联值&#xff08;Associated Values&#xff09;3. 关联值举例4. 原始值5. 隐式原始值&#xff08;Implicitly Assigned Raw Values&#xff09;6. 递归枚举&#xff08;Recursive Enumeration&#xff09;7. MemoryLayout Swift -…

解锁无限创意—MidjourneyAI绘画系统源码 支持AI智能会话+分销功能 对接ChatGPT+Midjourney接口 集成国内外众多AI大模型

在数字化浪潮汹涌的时代&#xff0c;人工智能已经成为推动社会进步的重要力量。而在艺术创作领域&#xff0c;MidjourneyAI绘画系统正以其强大的源码支持、智能会话功能、以及独特的分销模式&#xff0c;引领着智能艺术的新潮流。 分享一款MidjourneyAI绘画系统的源码&#xf…

leetcode多个测试用例之间相互影响导致提交失败

背景 在做一道easy题&#xff0c;二叉树的中序遍历&#xff0c;我提交的代码如下 from typing import (Optional,List )# Definition for a binary tree node. class TreeNode:def __init__(self, val0, leftNone, rightNone):self.val valself.left leftself.right right…

基础动态规划 - 过河卒

过河卒 兵从A点走到B点的所有路径方案&#xff0c;且不能经过 “马能吃棋子”的格子。 如果没有马&#xff0c;那么这道题就是一个简单的从A点走到B点的所有路径情况的简单动态规划。 状态转移方程为 dp[i,j] dp[i - 1,j] dp[i,j - 1]。 但如果加上了马这个棋子&#xff0…

一份报告实现两电平逆变、三电平逆变、三相整流、光伏并网simulink仿真

一份报告实现两电平逆变、三电平逆变、三相整流、光伏并网simulink仿真。逆变、整流与光伏的全家桶系列&#xff0c;适合小白使用。 模型获取链接&#xff1a;一份报告实现两电平逆变、三电平逆变、三相整流、光伏并网simulink仿真

llama3本地部署

目录 II.下载 II.验证ollama安装 II.安装llama3 和启动 II.命令行调用 II.api调用 II.参考文献 II.下载 https://ollama.com/download/windows OllamaSetup.exe https://github.com/meta-llama/llama3 II.验证ollama安装 cmd ollama II.安装llama3 和启动 ollama run …

LeetCode 2385.感染二叉树需要的总时间:两次搜索(深搜 + 广搜)

【LetMeFly】2385.感染二叉树需要的总时间&#xff1a;两次搜索&#xff08;深搜 广搜&#xff09; 力扣题目链接&#xff1a;https://leetcode.cn/problems/amount-of-time-for-binary-tree-to-be-infected/ 给你一棵二叉树的根节点 root &#xff0c;二叉树中节点的值 互不…

永磁同步电机SMO负载转矩观测matlab模型。

永磁同步电机SMO负载转矩观测matlab模型。 负载转矩的有效识别是提高伺服驱动系统抗负载扰动性能的关键之一。现在的传统结构的LTID滑模观测器存在频率抖动大&#xff0c;估计精度差的缺点&#xff0c;限制了其在高性能伺服系统中的应用。 本模型推导分析了传统LTID滑模观测器…

图片浏览工具-Honeyview

一、软件特点 轻量而快速 可以显示包括 GPS 信息在内的 JPEG 格式的 EXIF 信息 对图像格式进行批量转换和调整大小 支持显示 GIF 和 WebP 动图 无需解压即可直接查看压缩包中的图像 二、支持的格式 图像格式: BMP, JPG, GIF, PNG, PSD, DDS, JXR, WebP, J2K, JP2, TGA, TIFF, …

电商系统-农产品销售网站

一、今天给大家介绍一个VUESPRINGBOOT实现的农产品销售网站系统。包含商品海报轮播、商品分类展示、商品排行展示、推荐商品展示、购物车、订单、收藏、个人中心、收货地址维护、充值、如果是认证的商家还可以接入支付宝支付系统。下面来看下界面UI吧。 二、商品展示 三、购物车…

239.滑动窗口最大值

经典的单调队列的题目,不刷必后悔 思路解析: 显然我们可以可以写一个简单的O(nk)的算法,但是会超时.考虑到本题是需要维护一个有限制的最大值,我们使用单调队列这个数据结构优化这个过程. 单调队列里面存储元素的下标,队头是当前的窗口最大值,按照递减排列,显然从队头到队尾的…

聊聊Flink:Docker搭建Flink

一、准备工作 查看下Docker和Docker Compose版本&#xff0c;确保你安装了这些软件。 在 Flink 官网上下载 Flink 的 Docker 镜像。您可以使用以下命令从 Docker Hub 中下载&#xff1a; docker pull flink:1.18.0-scala_2.12 此命令将下载 Flink 1.18.0 版本的 Docker 镜像…

诡异的scp导致文件被清空问题

scp简介 scp 是 Linux 系统下基于 SSH 登陆进行安全的远程文件拷贝命令。它允许你在不同的 Linux 机器之间复制文件或目录。 使用 scp 命令的基本格式如下&#xff1a; 从本地复制到远程&#xff1a; scp [options] source_file userremote:destination_folder 从远程复制…

一文读懂Partisia Blockchain 的互操作方案:Oracle 服务框架

Partisia Blockchain 是一个以 MPC 技术为特点的创新隐私公链&#xff0c;其通过将 MPC 技术方案引入到区块链系统中&#xff0c;并以零知识证明&#xff08;ZK&#xff09;技术和多方计算&#xff08;MPC&#xff09;等方案为基础&#xff0c;共同保障在不影响网络完整性和安全…

面试八股——HashMap

实现原理 红黑树是为了解决链表过长之后&#xff0c;查找时间过长的问题&#xff0c;将链表存储变为红黑树存储。 put方法的实现&#xff08;5⭐&#xff09; 相关属性&#xff1a; 1. 容量&#xff1a;初始容量为2^4。 2. 加载因子&#xff1a;初始值为0.75 上面两个属性的…

GITEE本地项目上传到远程

由于需要&#xff0c;我这边将本地的仓库上传至GITEE。之前在网上搜索了相关的文档&#xff0c;但是步骤很繁琐&#xff0c;我这边介绍一个非常简单的。 一、在GITEE新建仓库 跟着指引一步步新建。 二、打开本地仓库&#xff0c;删除.git文件 默认情况下不会有这个.git文件&a…

李廉洋:4.29周一黄金原油走势分析,做单必看策略,

在关于美联储是否降息以及何时降息的争论中&#xff0c;另一场重要的争论正在展开&#xff1a;长期来看&#xff0c;利率将何去何从&#xff1f;问题的关键在于中性利率&#xff1a;即能够使储蓄供求平衡、同时保证经济增长和通胀稳定的利率。中性利率有时被称为“r*”或“r-st…

便携式iv检测仪解析

TH-PV31光伏电站便携式IV功率测试仪是一种专门用于光伏电站运维和故障排查的设备。它具备高精度、快速测试以及便携性等特点&#xff0c;成为光伏电站日常运维中不可或缺的工具。 首先&#xff0c;从工作原理来看&#xff0c;光伏电站便携式IV功率测试仪通过模拟太阳光照射光伏…

GaussianCube:使用最优传输构造高斯溅射用于3D生成建模

GaussianCube: Structuring Gaussian Splatting using Optimal Transport for 3D Generative Modeling GaussianCube&#xff1a;使用最优传输构造高斯溅射用于3D生成建模 Bowen Zhang1⁣*    Yiji Cheng2⁣*   Jiaolong Yang3   Chunyu Wang3 张博文 1⁣* 程一季 2⁣* …

第三节课,后端登录【1】.1--本人

一、后端登录逻辑&#xff0c;检测账户密码是否合法及密码输入是否正确 视频链接&#xff1a; 网址&#xff1a; 第三节&#xff1a;【视频】后端登录逻辑&#xff0c;检测账户密码是否合法及密码输入是否正确视频链接&#xff1a;-CSDN博客 从5.1开始 这是一个Java方法&am…