文件或资源的上传和下载

1.文件的上传

[1] 简介

> 将一个客户端的本地的文件发送到服务器中保存。

> 上传文件是通过流的形式将文件发送给服务器。

[2] 表单的设置

> 向服务器上传一个文件时,表单要使用post请求。

> 表单的默认属性enctype="application/x-www-form-urlencoded"

- 这个属性的意思是请求体中的内容将会使用URL编码

> 上传文件的表单enctype需要设置为 multipart/form-data

- multipart/form-data表示的是表单是一个多部件的表单

- 如果类型设置为它,则我们的每一个表单项都会作为一个单独的部件发送给服务器。

- 多个部件之间使用类似 -----------------------------7df2d08c0892 分割符来分开

> 当表单设置为multipart/form-data时,我们request.getParameter()将失效,我们不能再通过该方法获取请求参数。

[3] FileUpload

> 我们一般情况下使用commons-fileupload-1.3.1.jar这个工具来解析多部件请求。

> fileupload 依赖 commons-io 所以我们要是Filtupload需要同时导入io包。

> 核心类:

DiskFileItemFactory

- 工厂类,用于构建一个解析器实例。

ServletFileUpload

- 解析器类,通过该类实例来解析request中的请求信息。

FileItem

- 工具会将我们请求中每一个部件,都封装为一个FileItem对象,处理文件上传时,只需要调用该对象的方法

- 方法:

boolean isFormField()  --> 当前表单项是否是一个普通表单项,true是普通表单项, false是文件表单项

String getContentType() --> 返回的是文件的类型,是MIME值

String getFieldName()   --> 获取表单项的name属性值

String getName() --> 获取上传的文件的名字

long getSize()          --> 获取文件的大小

String getString(String encoding) --> 获取表单项的value属性值,需要接受一个编码作为参数。

void write(File file)    --> 将表单项中的内容写入到磁盘中

> 使用步骤:

1.获取工厂类实例[DiskFileItemFactory]

2.获取解析器类实例[ServletFileUpload]

3.解析request获取FileItem[parseRequest()]

[4] 细节

第一个问题

> 部分浏览器会将文件的完整路径作为文件名发送。

C:\Users\lilichao\Desktop\day20\图片\蒙娜丽莎.jpg

> 像这类文件名我们需要截取一下字符串,只获取名字这部分,而不需要获取路径部分的信息。

通过如下代码对文件名进行截取字符串的操作:

if(name.contains("\\")){
//如果包含则截取字符串
name = name.substring(name.lastIndexOf("\\")+1);
}

第二个问题

> 上传的文件有可能出现重名,后上传的文件会将先上传的文件覆盖。

> 解决:给文件名加一个唯一的前缀。

唯一标识_fennu.jpg

UUID_fennu.jpg

第三个问题

> 有些情况需要限制上传文件的大小。

- 设置单个文件大小为50KB:

fileUpload.setFileSizeMax(1024*50);

- 设置完单个文件大小限制以后,一旦上传的文件超过限制,则会抛出如下异常:

FileSizeLimitExceededException

所有可以对该异常进行捕获,当出现该异常时则设置一个错误消息。

- 设置多个文件的总大小为150KB

fileUpload.setSizeMax(1024*150);

- 当多个文件的大小超出范围时,会抛出如下异常

SizeLimitExceededException

第四个问题

> 当用户上传一个空的文件,依然会将文件保存到硬盘上。

> 在保存文件应该先对文件的大小进行判断,如果size为0,则不处理。

2.文件的下载

[1] 简介

> 将服务器中的文件下载到本地。

> 一般情况下资源所在的链接发送给浏览器,浏览器就会自动下载。

但是当浏览器支持当前文件的格式,浏览器会自动打开文件,而不会弹出下载窗口。

> 直接将资源放在项目的目录下,浏览器可以直接访问到资源。

所以一般我们下载的资源不能让浏览器直接访问到。

[2] 下载所需要的内容

1.获取到文件的流

2.设置两个响应头

[3] 下载相关的两个响应头

1) 文件类型 Content-Type --> 文件的MIME类型

Content-Type:告诉浏览器文件的类型,需要设置一个MIME值

response.setContent-Type("MIME值")

通过servletContext.getMimeType(path)方法可以直接获取文件的MIME类型

2) 下载文件的信息 Content-Disposition --> attachment; filename=文件名

Content-Disposition告诉浏览器如何处理文件,

attachment 告诉浏览器这个文件是一个附件的形式发给你的,需要你做下载的操作

filename 告诉浏览器下载文件的名字

3) 乱码的问题,当将文件的名字设置为中文,浏览器正常显示文件的名字。

因为从服务器向浏览器发送中文时,需要对内容进行URL编码。

> 大部分浏览器使用如下方式即可解决乱码问题:URLEncoder.encode(fileName, "utf-8");

> 但是火狐默认以Base64来解码的,所以要为火狐单独处理。

> 可以使用如下代码来判断浏览器的类型,然后进行不同的编码处理

 1 //判断当前浏览器是否为火狐
 2
 3 if(ua.contains("Firefox")){
 4
 5 //是火狐浏览器,使用BASE64编码
 6
 7 fileName = "=?utf-8?b?"+new BASE64Encoder().encode(fileName.getBytes("utf-8"))+"?=";
 8
 9 }else{
10
11 //给文件名进行URL编码
12
13 //URLEncoder.encode()需要两个参数,第一个参数时要编码的字符串,第二个是编码所采用的字符集
14
15 fileName = URLEncoder.encode(fileName, "utf-8");
16
17 }

>  还有一种不太讲理的方式,谁问跟谁急。反正好使

- 向将字符串用gbk进行解码,然后在使用iso8859-1进行编码

fileName = new String(fileName.getBytes("gbk"),"iso8859-1");

3.案例

[1]文件上传:

(1)原生方法实现文件上传

 1 public void testUpload(CommonsMultipartFile file,HttpServletRequest request){
 2 //声明输入流和输出流
 3 InputStream in=null;OutputStream out=null;
 4 //获取ServletContext
 5 ServletContext servletContext=request.getServletContext();
 6 //获取文件目录全路径
 7 String realPath = servletContext.getRealPath("/WEB-INF/upload");
 8 //获取文件名
 9 String fileName=file.getOriginalFilename();
10 //声明唯一标识,保证每次上传保存的文件名不同,从而使每次上传都可以成功,即使是相同的文件或资源
11 UUID id=UUID.randomUUID();
12 //将唯一标识与原文件名相加生成新资源文件名
13 fileName=id+"_"+fileName;
14 //通过文件目录全路径新建一个文件对象
15 File file1=new File(realPath);
16 //判断文件目录是否存在
17 if(!file1.exists()){
18 //如果不存在,则创建目录文件夹
19 file1.mkdirs();
20
21 }
22
23 try {
24 //设置输入流与输出流
25 in=file.getInputStream();
27 out=new FileOutputStream(new File(realPath+"\\"+fileName));
28 //设置缓冲区
29 byte[] buffer=new byte[1024];
30
31 int len=0;
32 //判断文件资源是否输入完,如果输入晚,则len==-1
33 while((len=in.read(buffer))!=-1){
34 //通过输出流写入文件
35 out.write(buffer, 0, len);
36
37 }
38 //这是简便方法直接可以将输入流写入输出流
39 /*IOUtils.copy(in, out);*/
40
41 } catch (IOException e) {
42
43 // TODO Auto-generated catch block
44
45 e.printStackTrace();
46
47 }finally{
48 //关闭输入流
49 if(in!=null){
50
51 try {
52
53 in.close();
54
55 } catch (IOException e) {
56
57 // TODO Auto-generated catch block
58
59 e.printStackTrace();
60
61 }
62
63 }
64 //关闭输出流
65 if(out!=null){
66
67 try {
68
69 out.close();
70
71 } catch (IOException e) {
72
73 // TODO Auto-generated catch block
74
75 e.printStackTrace();
76
77 }
78
79 }
80
81 }
82
83 }

(2)通过fileItem实现文件上传

 1 protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
 2
 3 DiskFileItemFactory dFactory=new DiskFileItemFactory();
 4
 5 ServletFileUpload sUpload=new ServletFileUpload(dFactory);
 6
 7 sUpload.setHeaderEncoding("UTF-8");
 8
 9 try {
10
11 sUpload.setFileSizeMax(1024*50);//设置单个上传文件的大小
12
13 sUpload.setSizeMax(1024*1024);//设置总上传文件的大小
14
15 List<FileItem> list = sUpload.parseRequest(request);
16
17 for(FileItem f:list){
18
19 if(f.isFormField()){
20
21 String name = f.getFieldName();
22
23 String string = f.getString();
24
25 System.out.println(name+":"+string);
26
27 }
28
29 else {
30
31  //获取文件名
32
33     String fileName = f.getName();
34
35     if(fileName.contains("\\")){
36
37      fileName = fileName.substring(fileName.lastIndexOf("\\")+1);
38
39     }
40
41     //获取上传路径
42
43     String realPath = getServletContext().getRealPath("/WEB-INF/upload");
44
45     //检查upload文件夹是否存在,如果不存在则创建
46
47     System.out.println(realPath);
48
49     File file = new File(realPath);
50
51     if(!file.exists()){
52
53     file.mkdirs();
54
55     };
56
57     //为避免重名生成一个uuid作为文件名的前缀
58
59     String prefix = UUID.randomUUID().toString().replace("-", "");
60
61     //将文件写入到服务器中
62
63     f.write(new File(realPath+"/"+prefix+"_"+fileName));
64
65     //清楚文件缓存
66
67     f.delete();
68
69 }
70
71 }
72
73 } catch(FileSizeLimitExceededException e){
74
75 System.out.println("文件内容太大,无法上传");
76
77 }catch ( Exception e) {
78
79 // TODO Auto-generated catch block
80
81 e.printStackTrace();
82 }
83 }

[2]文件下载:

protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

// TODO Auto-generated method stub

ServletContext servletContext=request.getServletContext();

String fileName="风吹麦浪.mp3";

String path = servletContext.getRealPath("WEB-INF/"+fileName);

File file=new File(path);

String type = servletContext.getMimeType(path);

InputStream inputStream=new FileInputStream(file);

response.setContentType(type);

fileName=new String(fileName.getBytes("gbk"),"iso8859-1");

response.setHeader("Content-Disposition","attachment;filename="+fileName);

ServletOutputStream outputStream=response.getOutputStream();

IOUtils.copy(inputStream, outputStream);

}
时间: 2024-10-14 22:23:08

文件或资源的上传和下载的相关文章

PHP之文件的锁定、上传与下载

小结文件的锁定机制.上传和下载 1.文件锁定 现在都在讲究什么分布式.并发等,实际上文件的操作也是并发的,在网络环境下,多个用户在同一时刻访问页面,对同一服务器上的同一文件进行着读取,如果,这个用户刚好读到一半,另一个用户就写入了消息,那么前一个用户读到的就是错误数据,在数据库里面好像是称为脏数据,而如果某用户写到一半时,另一用户也对该文件进行写操作,那么就造成了写入数据的混乱和错误,因此才php有一个锁机制,类似于数据库的锁,当某用户在对文件操作时就加上某种锁,使得在同一时间其他用户不能对该文

阿里云附件文件流的形式上传、下载

1.View附件上传 @using (Html.BeginForm("StartUpLoad", "ReferenceDocument", new {id = ViewBag.Id}, FormMethod.Post, new {enctype = "multipart/form-data"})) { <input type="file" name="filedata" id="fileUp

FTP上传和下载文件的应用

FTP(File Transfer Protocol)协议主要用来在网络上进行文件传输.FTP通讯除了有一个默认的端口21外,还有其他端口,同城两个端口同时进行数据传输.一个是默认的端口(通常为21),主要进行控制连接,即进行命令协议及服务器端响应码的传输.另一个非标准端口主要进行数据,上传下载文件等.具体上传.下载代码如下 import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundExcept

golang文件上传和下载

[代码]golang 实现的文件服务(包括上传,下载的server端和client端) (2013-09-20 02:03:52) 转载▼ 标签: golang go 文件服务器 it 分类: GO相关 //下载(支持断电续传)(client) package main import ( "http"     "os"     "io"     "strconv" ) const (     UA = "Golang

PHPstorm设置连接FTP,进行文件上传、下载、比较

                                             内容转载自:http://www.cnblogs.com/jikey/p/3486621.html   如何在Webstorm/Phpstorm中设置连接FTP, 并快速进行文件比较,上传下载,同步等操作 补充: 1. 打开Remote Host窗口的方式: Tools | Deployment | Browse Remote Host 或 View | Tool Windows | Remote Host

文件的上传与下载

为方便用户处理文件上传数据,Apache 开源组织提供了一个用来处理表单文件上传的一个开源组件( Commons-fileupload ),该组件性能优异,并且其API使用极其简单,可以让开发人员轻松实现web文件上传功能,因此在web开发中实现文件上传功能,通常使用Commons-fileupload组件实现. 使用Commons-fileupload组件实现文件上传,需要导入该组件相应的支撑jar包:Commons-fileupload和commons-io. commons-io 不属于文

JavaWeb学习总结(五十)——文件上传和下载

在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传和下载功能的实现. 对于文件上传,浏览器在上传的过程中是将文件以流的形式提交到服务器端的,如果直接使用Servlet获取上传文件的输入流然后再解析里面的请求参数是比较麻烦,所以一般选择采用apache的开源工具common-fileupload这个文件上传组件.这个common-fileupload上传组件的jar包可以去apache官网上面下载,也可以在struts的lib文件夹下面找到,stru

深入分析JavaWeb Item40 -- 文件上传和下载

在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传和下载功能的实现. 对于文件上传,浏览器在上传的过程中是将文件以流的形式提交到服务器端的,如果直接使用Servlet获取上传文件的输入流然后再解析里面的请求参数是比较麻烦,所以一般选择采用apache的开源工具common-fileupload这个文件上传组件.这个common-fileupload上传组件的jar包可以去apache官网上面下载,也可以在struts的lib文件夹下面找到,stru

JavaWeb文件上传和下载

在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传和下载功能的实现. 对于文件上传,浏览器在上传的过程中是将文件以流的形式提交到服务器端的,如果直接使用Servlet获取上传文件的输入流然后再解析里面的请 求参数是比较麻烦,所以一般选择采用apache的开源工具common-fileupload这个文件上传组件.这个common- fileupload上传组件的jar包可以去apache官网上面下载,也可以在struts的lib文件夹下面找到,st