Jersey中的上传下载

上传

在Jersey中的上传相对来说还是比较简单的。首先要了解的是,Jersey针对multipart需要专门的支持,首先在maven中添加multipart的依赖:

<dependency>
    <groupId>org.glassfish.jersey.media</groupId>
    <artifactId>jersey-media-multipart</artifactId>
    <version>2.25</version>
</dependency>

接着,需要在Jersey中引入MultipartFeature。MultiPartFeature是Jersey中针对Multipart的一种特征(Feature,Feature是JAX-RS中的一种规范,可以视为一种特殊的meta-provider,通常把多个相同类型的Provider封装到一起,比如针对多种encoding的Provider打包注册);
在Jersey中有几种注册MultipartFeature的方式:
1,在web.xml中注册:只需要在Jersey的ServletContainer中添加initparam即可:

<init-param>
        <param-name>jersey.config.server.provider.classnames</param-name>
        <param-value>org.glassfish.jersey.media.multipart.MultiPartFeature</param-value>
    </init-param>

2,如果是使用Servlet3.0的方式启动,需要在ResourceConfig中添加MultipartFeature:

@ApplicationPath("webapi")
public class RestApplication extends ResourceConfig {
    public RestApplication() {
        this.packages("cn.wolfcode.jersey");
        this.register(MultiPartFeature.class);
    }
}

配置完成之后,只需要完成对应的资源类即可:

@POST
@Path("image1")
@Produces(MediaType.APPLICATION_JSON)
@Consumes(MediaType.MULTIPART_FORM_DATA)
public String upload(@FormDataParam("file") InputStream fileInputStream,
        @FormDataParam("file") FormDataContentDisposition disposition, @Context ServletContext ctx) {
    
    return "success";
}

简单解释一下这段代码:
1,请求必须是POST的,这个不用多说;
2,请求返回json格式响应;
3,Consumes这次设置的是MediaType.MULTIPART_FORM_DATA,这个很重要,因为要能够上传,需要要求表单的提交格式为multipart/form-data;
4,重点在于资源方法的参数,在这里,我们使用了@FormDataParam(该标签来源于jersey的multipart包),该标签能够在资源方法上绑定请求编码类型为multipart/form-data中的每一个实体项。该标签可以绑定如下一些类型:

FormDataBodyPart;
List <FormDataBodyPart>;
FormDataContentDisposition;
List<FormDataContentDisposition>;
InputStream;

在介绍重点对象之前,我们先来简单回顾一下上传的基本概念。我们知道上传的时候,不能使用默认的x-www-form-urlencoded方式,而要选择使用multipart/form-data二进制传输方式,文件的内容才能够正常提交。那么使用Multipart/form-data方式提交,在请求实体中展示的内容如下:

image.png
其中,每一段内容(key:name,value:ssss和key:file,value:文件内容)就会被包装到一段content-disposition里面,用于代表本次提交的请求参数和请求文件内容;

明确请求格式之后,我们重点来关注三个对象:
1,FormDataBodyPart:对使用multipart/form-data编码请求实体的中的每一段实体内容的封装;提供了一些常见的方法:

//获取某一段实体请求的名字;
System.out.println(bp.getName());
//获取某一段实体请求的值(该方法只能作用于text/plain内容)
System.out.println(bp.getValue());
//获取某一段实体请求的封装对象,得到一个BodyPartEntity对象,可以调用getInputStream()方法获取实体内容等;
System.out.println(bp.getEntity());
//可以将请求转化为输入流(这也是一种获取上传文件内容的方法);
InputStream is=bp.getValueAs(InputStream.class);
//包装成FormDataContentDisposition对象返回;
FormDataContentDisposition disp=bp.getFormDataContentDisposition();

2,FormDataContentDisposition:对multipart/form-data中一段content-disposition的封装,较之FormDataBodyPart提供了更高级的方法抽象:

//获取文件名;
System.out.println("getFileName : " + disposition.getFileName());
//获取字段名称,即<input type="file" name="xxx")中的xxx
System.out.println("getName : " + disposition.getName());
//获取该段content-disposition的内容长度
System.out.println("getSize : " + disposition.getSize());
//获取该段content-disposition的类型,比如form-data
System.out.println("getType : " + disposition.getType());
//获取本次请求的请求值,比如{name=file, filename=3.jpg}
System.out.println("getParameters : " + disposition.getParameters());

3,InputStream:直接将提交的内容包装为输入流,这是最简单的上传文件方法;

介绍完几个重点对象,我们来看一个完整的上传文件的例子,使用两种方式完成:

@POST
@Path("image1")
@Produces(MediaType.APPLICATION_JSON)
@Consumes(MediaType.MULTIPART_FORM_DATA)
public String upload(@FormDataParam("file") InputStream fileInputStream,
        @FormDataParam("file") FormDataContentDisposition disposition, @Context ServletContext ctx) {

    File upload = new File(ctx.getRealPath("/upload"),
            UUID.randomUUID().toString() + "." + FilenameUtils.getExtension(disposition.getFileName()));
    try {
        FileUtils.copyInputStreamToFile(fileInputStream, upload);
    } catch (IOException e) {
        e.printStackTrace();
    }
    return "success";
}

第一种方式我们采用的是直接使用InputStream来提供上传文件的输入流;

@POST
@Path("image2")
@Produces(MediaType.APPLICATION_JSON)
@Consumes(MediaType.MULTIPART_FORM_DATA)
public String upload2(@FormDataParam("file") FormDataBodyPart bp, @Context ServletContext ctx) {
    File upload = new File(ctx.getRealPath("/upload"), UUID.randomUUID().toString() + "."
            + FilenameUtils.getExtension(bp.getFormDataContentDisposition().getFileName()));
    try {
        FileUtils.copyInputStreamToFile(bp.getValueAs(InputStream.class), upload);
    } catch (IOException e) {
        e.printStackTrace();
    }
    return "success";
}

第二种方式,我们使用较为底层的FormDataBodyPart来完成。综合两种方式,更建议使用第一种方式来完成文件上传。

下载

下载整体还是比较简单的操作,重点在于下载的相应(repsonse)的构造。所以在Jersey的下载演示中,我们重点在于介绍Jersey怎么使用Response对象来构造特殊的响应。

我们先来看一段基本的Jersey中用于下载的代码:

@GET
@Path("/images/{name}")
public void showImg(@PathParam("name") String imageName, @Context HttpServletResponse response,
        @Context ServletContext ctx) throws IOException {
    response.setHeader("Content-disposition", "attachment;filename="+imageName);
    response.setHeader("Cache-Control", "no-cache");
    File f = new File(ctx.getRealPath("/upload"), imageName);
    FileUtils.copyFile(f, response.getOutputStream());
}

可以看到,这段代码其实和以前使用Servlet来完成下载没有任何区别,原因就在于我们利用了Jersey的@Context标签可以注入HttpServletResponse对象的特性来完成的下载(注意,演示代码没有考虑到文件是否存在之类的情况);

在JAX-RS中,为我们提供了一个更为灵活的Response对象,通过建造模式(build pattern),能够非常方便的构造出适合的响应。我们先来看第一种使用Response完成下载的代码:

@GET
@Path("/images2/{name}")
public Response showImg(@PathParam("name") String imageName, @Context ServletContext ctx) throws IOException {
    File f = new File(ctx.getRealPath("/upload"), imageName);
    if (!f.exists()) {
        return Response.status(Status.NOT_FOUND).build();
    } else {
        return Response.ok(f).header("Content-disposition", "attachment;filename=" + imageName)
                .header("Cache-Control", "no-cache").build();
    }
}

这段代码关注几个点:
1,资源方法返回了一个javax.ws.rs.core.Response类,这个类可以直接被JAX-RS作为一个Response输出;
2,在代码中,如果文件找不到,我们直接使用Response类提供的status方法,构造了一个NOT_FOUND(404)状态码,并通过build方法创建一个Response对象返回,相当于就是构建了一个404状态码的响应。
3,如果文件存在,我们调用了Response的ok方法,传入了文件对象,那么JAX-RS会自动的设置本次响应状态码为200,并把文件写到响应输出流中,其次,我们采用链式调用的方式,设置了响应头,最后还是通过build方法,输出响应。总的来看,Response的这种操作方式,在开发上面还是非常方便和直观的。

这段代码演示了怎么下载一个文件,对于某些没有源文件,直接输出流的内容,比如在内存中构造一个图片提供下载,或者构造一个Excel下载或者构造一个PDF文件提供下载等,就不能直接把一个原始File对象作为ok方法的参数,这个时候就需要使用到JAX-RS提供的StreamingOutput接口:

@GET
@Path("/images3/{name}")
public Response showImg2(@PathParam("name") String imageName, @Context ServletContext ctx) throws IOException {
    final File f = new File(ctx.getRealPath("/upload"), imageName);
    if (!f.exists()) {
        return Response.status(Status.NOT_FOUND).build();
    } else {
        return Response.ok(new StreamingOutput() {
            @Override
            public void write(OutputStream output) throws IOException, WebApplicationException {
                output.write(FileUtils.readFileToByteArray(f));
            }
        }).header("Content-disposition", "attachment;filename=" + imageName)
          .header("Cache-Control", "no-cache").build();
    }
}

在这端代码中,重点在于Response对象的ok方法,我们传入了一个匿名内部类,实现了StreamingOutput接口,在方法write中,直接把byte内容输出到了output参数中,这个output即为response的输出流,即可。

关于Response对象的更多方法,可以参考其API文档,方法的名称大都是见名知意的,使用非常方便。

小结
本节简单的阐述了在Jersey中对文件的上传下载的基本操作,需要重点注意的是,要在Jersey中对上传进行操作,必须要引入MultipartFeature,才能针对multipart/form-data等特殊请求编码格式进行处理。

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值