将文件从 REST Web 服务发送到客户端的正确方法是啥?

Posted

技术标签:

【中文标题】将文件从 REST Web 服务发送到客户端的正确方法是啥?【英文标题】:what's the correct way to send a file from REST web service to client?将文件从 REST Web 服务发送到客户端的正确方法是什么? 【发布时间】:2012-08-27 17:15:35 【问题描述】:

我刚刚开始开发 REST 服务,但遇到了一个困难的情况:将文件从我的 REST 服务发送到我的客户端。到目前为止,我已经掌握了如何发送简单数据类型(字符串、整数等)的窍门,但是发送文件是另一回事,因为文件格式太多,我什至不知道应该从哪里开始。我的 REST 服务是在 Java 上创建的,我使用的是 Jersey,我使用 JSON 格式发送所有数据。

我读过关于 base64 编码的文章,有人说这是一种很好的技术,其他人说这不是因为文件大小问题。正确的方法是什么?这是我项目中一个简单资源类的外观:

import java.sql.SQLException;
import java.util.List;

import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.Context;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Request;
import javax.ws.rs.core.UriInfo;

import com.mx.ipn.escom.testerRest.dao.TemaDao;
import com.mx.ipn.escom.testerRest.modelo.Tema;

@Path("/temas")
public class TemaResource 

    @GET
    @Produces(MediaType.APPLICATION_JSON)
    public List<Tema> getTemas() throws SQLException

        TemaDao temaDao = new TemaDao();        
        List<Tema> temas=temaDao.getTemas();
        temaDao.terminarSesion();

        return temas;
    

我猜发送文件的代码是这样的:

import java.sql.SQLException;

import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;

@Path("/resourceFiles")
public class FileResource 

    @GET
    @Produces(application/x-octet-stream)
    public File getFiles() throws SQLException //I'm not really sure what kind of data type I should return

        // Code for encoding the file or just send it in a data stream, I really don't know what should be done here

        return file;
    

我应该使用什么样的注释?我看到有人推荐@GET 使用@Produces(application/x-octet-stream),这是正确的方法吗?我发送的文件是特定的,因此客户端不需要浏览这些文件。谁能指导我如何发送文件?我应该使用 base64 对其进行编码以将其作为 JSON 对象发送吗?还是不需要编码就可以将其作为 JSON 对象发送?感谢您提供的任何帮助。

【问题讨论】:

您的服务器上是否有实际的java.io.File(或文件路径),或者数据是否来自其他来源,如数据库、Web 服务、返回InputStream 的方法调用? 【参考方案1】:

由于您使用的是 JSON,我会在通过网络发送之前对其进行 Base64 编码。

如果文件很大,请尝试查看 BSON,或其他更适合二进制传输的格式。

如果文件压缩良好,您也可以在 base64 编码之前压缩文件。

【讨论】:

出于整个文件大小的原因,我打算在发送它们之前对其进行压缩,但如果我对其进行 base64 编码,我的 @Produces 注释应该包含什么? application/json 根据 JSON 规范,无论您放入什么。 (ietf.org/rfc/rfc4627.txt?number=4627) 请记住,base64 编码文件仍应位于 JSON 标记内 在 base64 中编码二进制数据然后将其包装在 JSON 中没有任何好处。它只会不必要地增加响应的大小并减慢速度。【参考方案2】:

如果你想返回一个要下载的文件,特别是如果你想与一些文件上传/下载的 javascript 库集成,那么下面的代码应该可以完成这项工作:

@GET
@Path("/key")
public Response download(@PathParam("key") String key,
                         @Context HttpServletResponse response) throws IOException 
    try 
        //Get your File or Object from wherever you want...
            //you can use the key parameter to indentify your file
            //otherwise it can be removed
        //let's say your file is called "object"
        response.setContentLength((int) object.getContentLength());
        response.setHeader("Content-Disposition", "attachment; filename="
                + object.getName());
        ServletOutputStream outStream = response.getOutputStream();
        byte[] bbuf = new byte[(int) object.getContentLength() + 1024];
        DataInputStream in = new DataInputStream(
                object.getDataInputStream());
        int length = 0;
        while ((in != null) && ((length = in.read(bbuf)) != -1)) 
            outStream.write(bbuf, 0, length);
        
        in.close();
        outStream.flush();
     catch (S3ServiceException e) 
        e.printStackTrace();
     catch (ServiceException e) 
        e.printStackTrace();
    
    return Response.ok().build();

【讨论】:

【参考方案3】:

我不建议使用 base64 编码二进制数据并将其包装在 JSON 中。它只会不必要地增加响应的大小并减慢速度。

使用 GET 和 application/octect-stream 使用 javax.ws.rs.core.Response 的一种工厂方法(JAX-RS API 的一部分,因此您不会被锁定在 Jersey)简单地提供您的文件数据:

@GET
@Produces(MediaType.APPLICATION_OCTET_STREAM)
public Response getFile() 
  File file = ... // Initialize this to the File path you want to serve.
  return Response.ok(file, MediaType.APPLICATION_OCTET_STREAM)
      .header("Content-Disposition", "attachment; filename=\"" + file.getName() + "\"" ) //optional
      .build();

如果您没有实际的 File 对象,但有一个 InputStreamResponse.ok(entity, mediaType) 也应该能够处理它。

【讨论】:

谢谢,这很好用,但是如果我想使用整个文件夹结构怎么办?我在想this 之类的东西,因为我将在客户端接收各种文件,我应该如何处理 HttpResponse 的实体响应? 查看ZipOutputStream 并从getFile() 返回StreamingOutput。通过这种方式,您可以获得大多数客户应该能够轻松读取的众所周知的多文件格式。仅在对您的数据有意义时才使用压缩,即不适用于 JPEG 等预压缩文件。在客户端,有ZipInputStream 来解析响应。 这可能会有所帮助:***.com/questions/10100936/… 有没有办法在响应中添加文件的元数据以及文件二进制数据? 您可以随时在响应中添加更多标头。如果这还不够,您必须将其编码到八位字节流中,即提供包含元数据和所需文件的容器格式。【参考方案4】:

将机器地址从 localhost 更改为您希望客户端连接的 IP 地址以调用下面提到的服务。

客户端调用 REST web 服务:

package in.india.client.downloadfiledemo;

import java.io.BufferedInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response.Status;

import com.sun.jersey.api.client.Client;
import com.sun.jersey.api.client.ClientHandlerException;
import com.sun.jersey.api.client.ClientResponse;
import com.sun.jersey.api.client.UniformInterfaceException;
import com.sun.jersey.api.client.WebResource;
import com.sun.jersey.multipart.BodyPart;
import com.sun.jersey.multipart.MultiPart;

public class DownloadFileClient 

    private static final String BASE_URI = "http://localhost:8080/DownloadFileDemo/services/downloadfile";

    public DownloadFileClient() 

        try 
            Client client = Client.create();
            WebResource objWebResource = client.resource(BASE_URI);
            ClientResponse response = objWebResource.path("/")
                    .type(MediaType.TEXT_html).get(ClientResponse.class);

            System.out.println("response : " + response);
            if (response.getStatus() == Status.OK.getStatusCode()
                    && response.hasEntity()) 
                MultiPart objMultiPart = response.getEntity(MultiPart.class);
                java.util.List<BodyPart> listBodyPart = objMultiPart
                        .getBodyParts();
                BodyPart filenameBodyPart = listBodyPart.get(0);
                BodyPart fileLengthBodyPart = listBodyPart.get(1);
                BodyPart fileBodyPart = listBodyPart.get(2);

                String filename = filenameBodyPart.getEntityAs(String.class);
                String fileLength = fileLengthBodyPart
                        .getEntityAs(String.class);
                File streamedFile = fileBodyPart.getEntityAs(File.class);

                BufferedInputStream objBufferedInputStream = new BufferedInputStream(
                        new FileInputStream(streamedFile));

                byte[] bytes = new byte[objBufferedInputStream.available()];

                objBufferedInputStream.read(bytes);

                String outFileName = "D:/"
                        + filename;
                System.out.println("File name is : " + filename
                        + " and length is : " + fileLength);
                FileOutputStream objFileOutputStream = new FileOutputStream(
                        outFileName);
                objFileOutputStream.write(bytes);
                objFileOutputStream.close();
                objBufferedInputStream.close();
                File receivedFile = new File(outFileName);
                System.out.print("Is the file size is same? :\t");
                System.out.println(Long.parseLong(fileLength) == receivedFile
                        .length());
            
         catch (UniformInterfaceException e) 
            e.printStackTrace();
         catch (ClientHandlerException e) 
            e.printStackTrace();
         catch (FileNotFoundException e) 
            e.printStackTrace();
         catch (IOException e) 
            e.printStackTrace();
        

    

    public static void main(String... args) 
        new DownloadFileClient();
    

响应客户端的服务:

package in.india.service.downloadfiledemo;

import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;

import com.sun.jersey.multipart.MultiPart;

@Path("downloadfile")
@Produces("multipart/mixed")
public class DownloadFileResource 

    @GET
    public Response getFile() 

        java.io.File objFile = new java.io.File(
                "D:/DanGilbert_2004-480p-en.mp4");
        MultiPart objMultiPart = new MultiPart();
        objMultiPart.type(new MediaType("multipart", "mixed"));
        objMultiPart
                .bodyPart(objFile.getName(), new MediaType("text", "plain"));
        objMultiPart.bodyPart("" + objFile.length(), new MediaType("text",
                "plain"));
        objMultiPart.bodyPart(objFile, new MediaType("multipart", "mixed"));

        return Response.ok(objMultiPart).build();

    

需要 JAR:

jersey-bundle-1.14.jar
jersey-multipart-1.14.jar
mimepull.jar

WEB.XML:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
    id="WebApp_ID" version="2.5">
    <display-name>DownloadFileDemo</display-name>
    <servlet>
        <display-name>JAX-RS REST Servlet</display-name>
        <servlet-name>JAX-RS REST Servlet</servlet-name>
        <servlet-class>com.sun.jersey.spi.container.servlet.ServletContainer</servlet-class>
        <init-param>
             <param-name>com.sun.jersey.config.property.packages</param-name> 
             <param-value>in.india.service.downloadfiledemo</param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>JAX-RS REST Servlet</servlet-name>
        <url-pattern>/services/*</url-pattern>
    </servlet-mapping>
    <welcome-file-list>
        <welcome-file>index.jsp</welcome-file>
    </welcome-file-list>
</web-app>

【讨论】:

以上是关于将文件从 REST Web 服务发送到客户端的正确方法是啥?的主要内容,如果未能解决你的问题,请参考以下文章

使用 REST API 进行正确的会话管理

将图像从服务器发送到客户端的专业方式

正确理解web交互中的cookie与session

套接字 io,节点 js,将图像/文件从服务器发送到客户端的简单示例

Camel Restlet异步发送对客户端的响应

使用 REST API 将图像从服务器发送到客户端