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

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

 以下是大致的工作流程

a1534bf0857f448b9e0492ddf1ac7ce4.png

 客户端有三个线程

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

内容如图:

9ba7370826884e3898999b1f42480627.png

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

8496d14c161b46b6bd73035ae4ffc3f9.png

 

客户端主线程代码:

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();
        }
    }
}

 

实验结果:

服务器

eb8343b5fdb04fd791d9e43231efb45d.png

 

 客户端:11e80ab5def148099c9813e87946f9b5.png

 

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

06291e72bfe248739b4de66d6e06d8f6.png

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

 8274394ba9f9477486bbba8f6d93db41.png

 

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

c7734d9c611140e8a64e144e47d49262.jpeg

 

 

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,这里提供一个简单的示例,演示如何使用Java的Socket编程实现客户送数据到服务器服务器接收到的数据返回给客户的流程。 客户代码: ```java import java.net.*; import java.io.*; public class Client { public static void main(String[] args) { String hostName = "127.0.0.1"; // 服务器IP int port = 8000; // 服务器口号 try (Socket socket = new Socket(hostName, port); PrintWriter out = new PrintWriter(socket.getOutputStream(), true); BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()))) { out.println("Hello, server!"); // 送消息给服务器 String response = in.readLine(); // 接收服务器返回的消息 System.out.println("Received from server: " + response); } catch (IOException e) { System.err.println("IOException: " + e.getMessage()); } } } ``` 在上面的代码,我们首先创建了一个Socket对象,并将其连接到指定的服务器地址和口号。然后,我们使用PrintWriter将要送的消息写入到输出流,并使用BufferedReader从输入流读取服务器返回的响应数据。最后,我们打印出接收到的数据。 服务代码: ```java import java.net.*; import java.io.*; public class Server { public static void main(String[] args) { int port = 8000; // 服务器口号 try (ServerSocket serverSocket = new ServerSocket(port); Socket clientSocket = serverSocket.accept(); PrintWriter out = new PrintWriter(clientSocket.getOutputStream(), true); BufferedReader in = new BufferedReader(new InputStreamReader(clientSocket.getInputStream()))) { System.out.println("Connected by " + clientSocket.getInetAddress().getHostAddress()); // 打印客户地址信息 String inputLine; while ((inputLine = in.readLine()) != null) { System.out.println("Received from client: " + inputLine); out.println(inputLine); // 将接收到的数据返回给客户 } } catch (IOException e) { System.err.println("IOException: " + e.getMessage()); } } } ``` 在上面的代码,我们首先创建了一个ServerSocket对象,并将其绑定到指定的服务器口号。然后,我们使用accept()方法开始监听客户的连接请求,并调用getInputStream()方法和getOutputStream()方法分别获取客户的输入流和输出流。在与客户建立连接后,我们使用BufferedReader从输入流读取客户送的数据,然后使用PrintWriter将接收到的数据返回给客户。 注意,在上面的代码,我们使用了一个while循环来不断地接收和处理客户的请求,直到客户主动关闭连接。 综上所述,以上代码演示了如何使用Java的Socket编程实现客户送数据到服务器服务器接收到的数据返回给客户的流程。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值