Java模拟文件发送给服务器,服务器将文件转发给其他用户,并保存到服务器本地,其他用户可以接收,并保存到本地磁盘,支持各种文件格式,并解决通信中服务器怎么区分客户端发来的文件类型

简介: Java模拟文件发送给服务器,服务器将文件转发给其他用户,并保存到服务器本地,其他用户可以接收,并保存到本地磁盘,支持各种文件格式,并解决通信中服务器怎么区分客户端发来的文件类型

在局域网中,客户可以将文件分享到网络上,由服务器进行转发给其他客户,其他客户可以接收服务器发来的文件,并保存到本地磁盘中。

以下是大致的工作流程

客户端有三个线程

主线程  1.负责启动 文件发送的线程        2.负责启动文件接收线程

内容如图:

服务端有两个线程,主线程专门接收用的连接,并为每一个连接上服务器的客户创建一个子线程。子线程专门用来接收客户发来的文件,保存本地磁盘,并转发给其他客户。如下:

 

客户端主线程代码:

import java.net.Socket;
 
//客户端
public class Client {
    public static void main(String[] args) {
        try {
            System.out.println("=======客户端=======");
            //1.线连接上服务器的套接字 (套接字=ip:端口号)
            Socket socket = new Socket("127.0.0.1", 10001);
 
            //2. 开启一个线程对象 专们用来接收文件的线程
            new ClientGetFile(socket).start();
 
            //3.开启发送文件的线程对象
            new  ClientSendFile(socket).start();
            while (true);
        } catch (Exception e) {
 
            e.printStackTrace();
        }
    }
}

客户端接收文件子线程代码:

import java.io.*;
import java.net.Socket;
//作为客户端接收文件的线程
public class ClientGetFile extends Thread{
    private final Socket socket;
 
    public ClientGetFile(Socket socket) {
        this.socket = socket;
    }
    @Override
    public void run() {
        try {
            //1.定义 输入流 从网络介质中获取数据存入内存中
            DataInputStream dis=new DataInputStream(socket.getInputStream());
 
            //2. 定义流将文件数据从内存中写入磁盘
            String FileName = dis.readUTF();
            String FilePath ="E:\\Documets\\Desktop\\客户\\"+ FileName;
 
            System.out.println("正在保存:"+FileName);
            DataOutputStream outputToDisk = new DataOutputStream(new FileOutputStream(FilePath));
            byte[] buffer=new byte[8192];
            long FileLength = dis.readLong();
            int length;
            int OKLength=0;
 
            while ((length=dis.read(buffer))!=-1){
                OKLength += length;
                outputToDisk.write(buffer,0,  length);
                outputToDisk.flush();
                if (FileLength==OKLength)
                    break;
            }
            System.out.println("接收"+FileName+"成功!");
            System.out.println("路径为:"+FilePath);
            outputToDisk.close();       //关闭 内存->磁盘 的io资源
        } catch (Exception e) {
                System.out.println("您已离线!!!");
                e.printStackTrace();
        }
    }
}

客户端发送文件子线程代码:

import java.io.*;
import java.net.Socket;
//作为客户端发送文件的线程类
public class ClientSendFile extends Thread{
    private final Socket socket;
 
    public ClientSendFile(Socket socket) {
        this.socket = socket;
    }
    @Override
    public void run() {
        try {
            //1.创建要发送的文件对象
            File file=new File("E:\\Documets\\Desktop\\老婆.mp4");
 
            //2.通过套接字传文件给服务器,由服务器转发给别的客户
            sendFileToServer(socket,file);
            System.out.println(file.getName()+"发送完毕!!!");
            //while (true);
        } catch (Exception e) {
            System.out.println("您已离线!!");
            e.printStackTrace();
        }
    }
    //定义一个静态方法 作为专门发送文件
    private static void sendFileToServer(Socket socket,File file) throws Exception {
        //1.将文件对象输入到内存中来
        DataInputStream InputToRAM=new DataInputStream( new FileInputStream(file) );
 
        //2.准备发送管道 发送到网络 给服务器接收
        DataOutputStream outputToNet= new DataOutputStream( socket.getOutputStream() );
 
        //3.发送文件名、文件大小 给服务器
        outputToNet.writeUTF(file.getName());   //发送文件名给 服务器   file.getName(); 得到要发送的文件名
        outputToNet.flush();                    //刷新流
        outputToNet.writeLong(file.length());  // 发送文件大小给 服务器   file.length();得到要发送的文件大小 单位字节(1K=1024Byte)
        outputToNet.flush();
 
        // 4.发送文件内容 给服务器
        int length;
        byte[] buffer = new byte[1024];
        while ((length = InputToRAM.read(buffer)) != -1) {  //dis.read(buffer)  从磁盘中读取内容,存到buffer数组中(数组在内存中)
            outputToNet.write(buffer,0,length);
            outputToNet.flush();
        }
    }
}

=========================================================================

服务器主线程代码:

import java.net.ServerSocket;
import java.net.Socket;
import java.util.ArrayList;
import java.util.List;
 
//使用两个线程 主线程 用于将连接过来的套接字 添加到集合 代表当前在线的人数
public class Server {
    /*创建一个Socket的list集合 用来装套接字  Socket=(IP地址:端口号)   */
    public static final List<Socket> onLineSockets = new ArrayList<>();  //当客户端连接上服务器的时候,就将客户端的套接字存入集合中
    public static void main(String[] args) {
        try {
            System.out.println("===服务端启动成功===");
            // 1、注册端口
            ServerSocket serverSocket = new ServerSocket(10001);
            // a.定义一个死循环由主线程负责不断的接收客户端的Socket管道连接。
            while (true) {
                // 2、每接收到一个客户端的Socket管道,交给一个独立的子线程负责读取消息
                Socket socket = serverSocket.accept();
                System.out.println(socket.getRemoteSocketAddress()+ "上线了!");
                onLineSockets.add(socket);// 把当前客户端管道Socket加入到在线集合中去
                // 3、开始创建独立线程处理这个连接上来的客户
                new ServerFileThread(socket).start();
            }
        } catch (Exception e) {
            System.out.println("您已离线!");
            e.printStackTrace();
        }
    }
}

服务器子线程,负责文件的保存和转发:

import java.io.*;
import java.net.Socket;
//服务端转发文件给客户线程
public class ServerFileThread extends Thread{
    private final Socket socket;                      //用来存套接字
    public ServerFileThread(Socket socket){   //构造方法,用来接收  与服务器连接的管道对管道进行读操作
        this.socket = socket;
    }
    @Override
    public void run() {
        try {
            //1.得到客户的通信管道
            DataInputStream InputToRAM=new DataInputStream(socket.getInputStream());
 
            //2.准备输出流  一个输出到服务器的本地磁盘下,一个输出到网络介质上,让其他客户接收
            DataOutputStream outputToDisk;
            DataOutputStream outputToNet = new DataOutputStream(socket.getOutputStream());
 
            //3. 接收和转发文件    (接收是接收到服务器的磁盘下,转发是转发到 网络介质上给别的客户)
            while (true) {
                // 获取文件名字和文件长度
                String FileName = InputToRAM.readUTF();
                String FilePaht ="E:\\Documets\\Desktop\\服务器\\"+ FileName;
                System.out.println("正在接收:"+FileName);
                outputToDisk=new DataOutputStream(new FileOutputStream(FilePaht));
                long FileLength = InputToRAM.readLong();
 
                // 发送文件名字和文件长度给所有客户端
                for(Socket onLineSocket : Server.onLineSockets) { //onLineSockets存的是当前连接的客户
                    if(onLineSocket!= socket) {       // 发送给其它客户端
                        outputToNet.writeUTF(FileName);
                        outputToNet.flush();
                        outputToNet.writeLong(FileLength);
                        outputToNet.flush();
                    }
                }
 
                //真正传送文件数据
                int length;
                long OKLength = 0;
                byte[] buffer = new byte[8192];
                while ((length = InputToRAM.read(buffer)) != -1) {
                    OKLength += length;        //记录已经传输的文件大小
                    //存到服务器的磁盘下
                    outputToDisk.write(buffer, 0, length);
                    outputToDisk.flush();
                    //转发数据到每个用户
                    for(Socket onLineSocket : Server.onLineSockets) {
                        if(onLineSocket != socket) {  // 发送给其它客户端,
                            outputToNet.write(buffer, 0, length);
                            outputToNet.flush();
                        }
                    }
                    if(OKLength == FileLength) {  // 强制退出
                        break;
                    }
                }
                System.out.println(FileName+"转发完毕!");
                System.out.println(FileName+"保存到服务器的路径为:"+FilePaht);
                outputToDisk.close();           //关闭 内存->磁盘 的io资源
            }
        } catch (IOException e) {
            Server.onLineSockets.remove(socket);
            System.out.println(socket.getRemoteSocketAddress()+"已下线~");     //该用户已经断开连接
            System.out.println("当前在线人数:"+Server.onLineSockets.size());
           // e.printStackTrace();
        }
    }
}

 

实验结果:

服务器

 

客户端:

 

在E:\Documets\Desktop\服务器\下有 “老婆.mp4” 文件,且可以正常打开

在E:\Documets\Desktop\客户\路径下也有,并可以正常打开

 

再来说说怎么解决 服务器怎么区别,客户发来的文件 是什么类型的文件,解决的办法是先将问文件名和文件大小发送给服务器,然后让服务器分配好io输出到本地磁盘的路径,之后再进行文件数据的传输,这样就可以了,详情参考上面的代码。

相关文章
|
4天前
|
Java
Java开发如何实现文件的移动,但是在移动结束后才进行读取?
【10月更文挑战第13天】Java开发如何实现文件的移动,但是在移动结束后才进行读取?
13 2
|
4天前
|
Java Apache Maven
Java将word文档转换成pdf文件的方法?
【10月更文挑战第13天】Java将word文档转换成pdf文件的方法?
12 1
|
4天前
|
监控 Java
Java定时扫码一个文件夹下的文件,如何保证文件写入完成后才进行处理?
【10月更文挑战第13天】Java定时扫码一个文件夹下的文件,如何保证文件写入完成后才进行处理?
16 1
|
8天前
|
网络协议 Unix Linux
一个.NET开源、快速、低延迟的异步套接字服务器和客户端库
一个.NET开源、快速、低延迟的异步套接字服务器和客户端库
|
8天前
|
运维 Java Linux
【运维基础知识】Linux服务器下手写启停Java程序脚本start.sh stop.sh及详细说明
### 启动Java程序脚本 `start.sh` 此脚本用于启动一个Java程序,设置JVM字符集为GBK,最大堆内存为3000M,并将程序的日志输出到`output.log`文件中,同时在后台运行。 ### 停止Java程序脚本 `stop.sh` 此脚本用于停止指定名称的服务(如`QuoteServer`),通过查找并终止该服务的Java进程,输出操作结果以确认是否成功。
14 1
|
8天前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
21 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
5天前
|
安全 区块链 数据库
|
8天前
|
存储 监控 Linux
充分利用服务器的磁盘资源,提高系统的稳定性和可维护性
充分利用服务器的磁盘资源,提高系统的稳定性和可维护性
19 0
|
Java 关系型数据库 图形学
Java获取系统文件类型图标并显示在JSP上
在网站制作中通常需要上传附件,而对于附件我们往往希望在其名称前面有类似于Windows系统中的类型图标,那么怎么根据附件的类型来显示不同的图标呢?目前有两种解决方案:  第一种:将所有类型文件的图标图片放置到项目中,然后通过分析文件的扩展名来调用相应的图片,这种方式比较简单常见,但是我们往往无法弄到所有文件类型的图标,而且也不能排除意外情况的出现,这里就不在介绍了;  第二种:通过java调用
1147 0
|
4天前
|
安全 Java UED
Java中的多线程编程:从基础到实践
本文深入探讨了Java中的多线程编程,包括线程的创建、生命周期管理以及同步机制。通过实例展示了如何使用Thread类和Runnable接口来创建线程,讨论了线程安全问题及解决策略,如使用synchronized关键字和ReentrantLock类。文章还涵盖了线程间通信的方式,包括wait()、notify()和notifyAll()方法,以及如何避免死锁。此外,还介绍了高级并发工具如CountDownLatch和CyclicBarrier的使用方法。通过综合运用这些技术,可以有效提高多线程程序的性能和可靠性。