Java:在套接字编程中从InputStream接收不同类型的文件

时间:2012-04-04 12:50:15

标签: java sockets file-io

在我的项目客户端中,将一些文本,一些图像发送到服务器。服务器需要处理所有这些事情,即如果有文本,那么它需要显示在框架上的TestArea上,如果图像文件在那里,则服务器需要将该图像文件存储在计算机上。 我创建了一个处理字符串(文本)或图像文件的应用程序,但是当客户端同时发送这些文件时,我不知道如何存储所有这些内容。

我知道添加此代码的时间: InputStream in = socket.getInputStream();

然后客户端发送的所有数据都在此 InputStream 中 所以我如何识别这么多范围的数据是图像文件所以需要存储在字节数组中,这要多得多是测试或字符串所以需要在TextAreea中显示。 如果客户端一次向服务器发送两个或多个图像,那么服务器如何理解这么多数据这是第一个图像,直到这么多数据这是第二个图像文件。

我尝试使用此代码发送图片:

客户代码:

public void sendPhotoToServer(String str){ // str is image location
try {
    InputStream input = new FileInputStream(str);
    byte[] buffer=new byte[1024];
    int readData;
    while((readData=input.read(buffer))!=-1){
    dos.write(buffer,0,readData); // dos is DataOutputStream
    }
} catch (FileNotFoundException e) {

} catch (IOException e) {

}       
}

并且此方法在循环中,因此客户端发送其文件夹中的所有图像。

现在服务器代码,这是在线程和while循环是每次监听客户端数据:

public void run() {
while (true) {
        try {
            byte[] buffer = new byte[8192];
            fis = new FileOutputStream("C:\\"+(s1++)+".jpg"); // fis is FileOutputStream
            while ((count = in.read(buffer)) > 0){ //count is a integer and 'in' is InputStream
            fis.write(buffer, 0, count); 
            fis.flush();    
            }
            } catch (Exception e) {}
}
}

但是通过此代码,服务器接收到第一张图像,之后它不显示其他图像。

以及服务器如何理解客户端是否发送简单字符串。 在服务器端应用程序中,它打开一个端口来监听客户端发送的所有数据

FileOutputStream fis = socket.getInputStream();

现在它如何区分所有这些文件和简单的字符串。

3 个答案:

答案 0 :(得分:2)

没有自动处理数据边界。服务器代码无法知道一个文件/消息的结束位置和另一个文件/消息的开始位置。你必须提出自己的协议。

答案 1 :(得分:2)

如果您事先知道数据类型,为什么不使用ObjectInputStreamObjectOutputStream

这里发送:

import java.io.*;

public class Client {

  public static void main(String[] args) throws IOException {
    // write string
    ObjectOutputStream out = new ObjectOutputStream(
      new BufferedOutputStream(
        new FileOutputStream("test-string.data")));
    try {
      out.writeObject("Hello");
    } finally {
      out.close();
    }
    // write byte arrays
    out = new ObjectOutputStream(
      new BufferedOutputStream(
        new FileOutputStream("test-byteArrays.data")));
    try {
      out.writeObject(new byte[] { 'H', 'e', 'l', 'l', 'o' });
      out.writeObject(new byte[] { 'W', 'o', 'r', 'l', 'd' });
    } finally {
      out.close();
    }
  }

}

在这里接收:

import java.io.*;
import java.util.*;

public class Server {

  public static void main(String[] args) throws IOException, ClassNotFoundException {
    // write string
    ObjectInputStream in = new ObjectInputStream(
      new BufferedInputStream(
        new FileInputStream("test-string.data")));
    try {
      Object o = null;
      while ((o = in.readObject()) != null) {
        System.out.printf("Class: %s, toString: %s\n", o.getClass(), o.toString());
      }
    } catch (EOFException e) {
      // finished
    } finally {
      in.close();
    }
    // write byte arrays
    in = new ObjectInputStream(
      new BufferedInputStream(
        new FileInputStream("test-byteArrays.data")));
    try {
      Object o = null;
      while ((o = in.readObject()) != null) {
        System.out.printf("Class: %s, toString: %s\n", o.getClass(), o.toString());
      }
    } catch (EOFException e) {
      // finished
    } finally {
      in.close();
    }
  }

}

答案 2 :(得分:1)

您正在做的是通过套接字发送原始数据。这就是你所拥有的:一个字节流。为了确保接收端能够理解哪个部分属于哪个文件,以及该文件的类型,您需要想出一种方法将其编码为字节流,以便实际数据没有受到干扰,发送者/接收者都“理解”这种结构。这就是所谓的协议。

在某些情况下,使用自己的协议可能很有用,但要知道根据您想要实现的目标,它可能非常复杂。很可能已经存在一个非常适合您需求的协议。您可以使用HTTP与标头字段中指示的MIME类型。这可以让你弄清楚文件类型,在文本的情况下,你可以找出编码。 FTP适用于文件传输,文件名的扩展名可用于检测其类型。 SOAP允许您以独立于客户端或服务器实现语言的方式发送复杂的数据结构。

TL; DR版本:不要重新发明轮子,搜索那里存在的东西以满足您的需求。最有可能的是,合适的协议支持Java开箱即用或通过第三方库。 Apache的Commons Net可以帮助你。