1

在我的项目客户端中,向服务器发送一些文本、一些图像。服务器需要处理所有这些事情,即如果存在文本,则需要在框架上的 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();

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

4

3 回答 3

2

如果您事先知道数据类型,为什么不直接使用ObjectInputStreamand 。ObjectOutputStream

这里是发送:

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

}
于 2012-04-04T13:54:20.390 回答
2

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

于 2012-04-04T13:02:16.060 回答
1

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

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

TL;DR 版本:不要重新发明轮子,四处寻找现有的东西以满足您的需求。最有可能的是,合适的协议支持开箱即用的 Java 或通过第三方库。Apache 的 Commons Net 可以在这里为您提供帮助。

于 2012-04-04T13:30:35.387 回答