一、文件上传的原理

    1、文件上传的前提:

        a、form表单的method必须是post

        b、form表单的enctype必须是multipart/form-data(决定了POST请求方式,请求正文的数据类型)

            注意:当表单的enctype是multipart/form-data,传统的获取请求參数的方法失效。

请求正文:(MIME协议进行描写叙述的,正文是多部分组成的)

            -----------------------------7dd32c39803b2

            Content-Disposition: form-data; name="username"



            wzhting

            -----------------------------7dd32c39803b2

            Content-Disposition: form-data; name="f1"; filename="C:\Documents and Settings\wzhting\妗岄潰\a.txt"

            Content-Type: text/plain



            aaaaaaaaaaaaaaaaaa

            -----------------------------7dd32c39803b2

            Content-Disposition: form-data; name="f2"; filename="C:\Documents and Settings\wzhting\妗岄潰\b.txt"

            Content-Type: text/plain



            bbbbbbbbbbbbbbbbbbb

            -----------------------------7dd32c39803b2--



            

        c、form中提供input的type是file类型的文件上传域

        

二、利用第三方组件实现文件上传

    1、commons-fileupload组件:

        jar:commons-fileupload.jar  commons-io.jar

    2、核心类或接口

        DiskFileItemFactory:设置环境

            public void setSizeThreshold(int?sizeThreshold) :设置缓冲区大小。

默认是10Kb。

                    当上传的文件超出了缓冲区大小。fileupload组件将使用暂时文件缓存上传文件

            public void setRepository(java.io.File repository):设置暂时文件的存放文件夹。

默认是系统的暂时文件存放文件夹。

ServletFileUpload:核心上传类(主要作用:解析请求的正文内容)

            boolean isMultipartContent(HttpServletRequest?

request):推断用户的表单的enctype是否是multipart/form-data类型的。

            List parseRequest(HttpServletRequest request):解析请求正文中的内容

            setFileSizeMax(4*1024*1024);//设置单个上传文件的大小

            upload.setSizeMax(6*1024*1024);//设置总文件大小

        FileItem:代表表单中的一个输入域。

boolean isFormField():是否是普通字段

            String getFieldName:获取普通字段的字段名

            String getString():获取普通字段的值

            

            InputStream getInputStream():获取上传字段的输入流

            String getName():获取上传的文件名称

            

三、文件上传中要注意的9个问题

    1、怎样保证server的安全

        把保存上传文件的文件夹放到WEB-INF文件夹中。

    2、中文乱码问题

        2.1普通字段的中文请求參数

            String value = FileItem.getString("UTF-8");

        2.2上传的文件名称是中文

            解决的方法:request.setCharacterEncoding("UTF-8");

    3、重名文件被覆盖的问题

            System.currentMillions()+"_"+a.txt(乐观)

            

            UUID+"_"+a.txt:保证文件名称唯一

    4、分文件夹存储上传的文件

        方式一:当前日期建立一个目录。当前上传的文件都放到此目录中。

方式二:利用文件名称的hash码打散文件夹来存储。

            int hashCode = fileName.hashCode();

            

                1001 1010 1101 0010 1101 1100 1101 1010

hashCode&0xf;   0000 0000 0000 0000 0000 0000 0000 1111 &

                ---------------------------------------------

                0000 0000 0000 0000 0000 0000 0000 1010   取hashCode的后4位

                                                    0000~1111:整数0~15共16个

                

                1001 1010 1101 0010 1101 1100 1101 1010

(hashCode&0xf0)    0000 0000 0000 0000 0000 0000 1111 0000  &

                --------------------------------------------

                0000 0000 0000 0000 0000 0000 1101 0000  >>4

                --------------------------------------------

                0000 0000 0000 0000 0000 0000 0000 1101

                                                    0000~1111:整数0~15共16个

    5、限制用户上传的文件类型

        通过推断文件的扩展名来限制是不可取的。

        通过推断其Mime类型才靠谱。

FileItem.getContentType();

    6、怎样限制用户上传文件的大小

        6.1单个文件限制大小。超出了大小友好提示

            抓异常进行提示:org.apache.commons.fileupload.FileUploadBase.FileSizeLimitExceededException

        6.2总文件限制大小。超出了大小友好提示

            抓异常进行提示:org.apache.commons.fileupload.FileUploadBase.SizeLimitExceededException

    7、暂时文件的问题

        commons-fileupload组件不会删除超出缓存的暂时文件。

        

        FileItem.delete()方法删除暂时文件。但一定要在关闭流之后。

8、多个文件上传时。没有上传内容的问题

        if(fileName==null||"".equals(fileName.trim())){

            continue;

        }

    9、上传进度检測

        给ServletFileUpload注冊一个进度监听器就可以。把上传进度传递给页面去显示

        //pBytesRead:当前以读取到的字节数

        //pContentLength:文件的长度

        //pItems:第几项

        public void update(long pBytesRead, long pContentLength,

                int pItems) {

            System.out.println("已读取:"+pBytesRead+",文件大小:"+pContentLength+",第几项:"+pItems);

}

四:文件上传演示样例:

JSP代码:

<form action="${pageContext.request.contextPath}/servlet/UploadServlet" method="post" enctype="multipart/form-data">
用户名:<input type="text" name="username"/><br/>
文件1:<input type="file" name="f1"/><br/>
文件2:<input type="file" name="f2"/><br/>
<input type="submit" value="保存"/>
</form>

servlet后台代码:

public class UploadServlet extends HttpServlet {

	public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
request.setCharacterEncoding("UTF-8");
response.setContentType("text/html;charset=UTF-8");
PrintWriter pout = response.getWriter();
try {
// 得到存放上传文件的真实路径
String storePath = getServletContext()
.getRealPath("/WEB-INF/files"); // 设置环境
DiskFileItemFactory factory = new DiskFileItemFactory();
factory.setRepository(new File(getServletContext().getRealPath("/temp")));//设置暂时存放文件夹
// 推断一下form是否是enctype=multipart/form-data类型的
boolean isMultipart = ServletFileUpload.isMultipartContent(request);
if (!isMultipart) {
System.out.println("大傻鸟");
return;
}
// ServletFileUpload核心类
ServletFileUpload upload = new ServletFileUpload(factory);
// upload.setProgressListener(new ProgressListener() {
// //pBytesRead:当前以读取到的字节数
// //pContentLength:文件的长度
// //pItems:第几项
// public void update(long pBytesRead, long pContentLength,
// int pItems) {
// System.out.println("已读取:"+pBytesRead+",文件大小:"+pContentLength+",第几项:"+pItems);
// }
//
// });
upload.setFileSizeMax(4 * 1024 * 1024);// 设置单个上传文件的大小
upload.setSizeMax(6 * 1024 * 1024);// 设置总文件大小
// 解析
List<FileItem> items = upload.parseRequest(request);
for (FileItem item : items) {
if (item.isFormField()) {
// 普通字段
String fieldName = item.getFieldName();
String fieldValue = item.getString("UTF-8");
System.out.println(fieldName + "=" + fieldValue);
} else {
// 得到MIME类型
String mimeType = item.getContentType();
// 仅仅同意上传图片
if(mimeType.startsWith("image")){
// 上传字段
InputStream in = item.getInputStream();
// 上传的文件名称
String fileName = item.getName();// C:\Documents and
if(fileName==null||"".equals(fileName.trim())){
continue;
}
// Settings\wzhting\妗岄潰\a.txt
// a.txt
fileName = fileName
.substring(fileName.lastIndexOf("\\") + 1);// a.txt
fileName = UUID.randomUUID() + "_" + fileName;
System.out.println(request.getRemoteAddr()+"=============="+fileName);
// 构建输出流
// 打散存储文件夹
String newStorePath = makeStorePath(storePath, fileName);// 依据
// /WEB-INF/files和文件名称,创建一个新的存储路径
// /WEB-INF/files/1/12
String storeFile = newStorePath + "\\" + fileName;// WEB-INF/files/1/2/sldfdslf.txt OutputStream out = new FileOutputStream(storeFile); byte b[] = new byte[1024];
int len = -1;
while ((len = in.read(b)) != -1) {
out.write(b, 0, len);
}
out.close();
in.close();
item.delete();//删除暂时文件
}
}
}
} catch (org.apache.commons.fileupload.FileUploadBase.FileSizeLimitExceededException e) {
// 单个文件超出大小时的异常
pout.write("单个文件大小不能超出4M");
} catch (org.apache.commons.fileupload.FileUploadBase.SizeLimitExceededException e) {
// 总文件超出大小时的异常
pout.write("总文件大小不能超出6M");
} catch (Exception e) {
e.printStackTrace();
} } // 依据 /WEB-INF/files和文件名称,创建一个新的存储路径 /WEB-INF/files/1/12
private String makeStorePath(String storePath, String fileName) {
int hashCode = fileName.hashCode();
int dir1 = hashCode & 0xf;// 0000~1111:整数0~15共16个
int dir2 = (hashCode & 0xf0) >> 4;// 0000~1111:整数0~15共16个 String path = storePath + "\\" + dir1 + "\\" + dir2; // WEB-INF/files/1/12
File file = new File(path);
if (!file.exists())
file.mkdirs(); return path;
} public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException { doGet(request, response);
} }

---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

五:文件下载

1.显示上述存储全部的照片信息:

//显示全部上传的文件。封装到域对象中,交给jsp去显示
public class ShowAllFilesServlet extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
Map<String, String> map = new HashMap<String, String>();//key:UUID文件名称;value:老文件名称
//得到存储文件的根文件夹
String storePath = getServletContext().getRealPath("/WEB-INF/files");
//递归遍历当中文件
File file = new File(storePath);
treeWalk(file,map);
//交给JSP去显示:怎样封装数据.用Map封装。key:UUID文件名称。value:老文件名称
request.setAttribute("map", map);
request.getRequestDispatcher("/listFiles.jsp").forward(request, response);
}
//遍历/WEB-INF/files全部文件,把文件名称放到map中
private void treeWalk(File file, Map<String, String> map) {
if(file.isFile()){
//是文件
String uuidName = file.getName();// UUID_a_a.txt//真实文件名称
String oldName = uuidName.substring(uuidName.indexOf("_")+1);
map.put(uuidName, oldName);
}else{
//是一个文件夹
File[] fs = file.listFiles();
for(File f:fs){
treeWalk(f,map);
}
}
} public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException { doGet(request, response);
} }

2.显示全部文件listFiles.jsp页面中的内容:

<body>
<h1>本站有下面好图片</h1>
<c:forEach items="${map}" var="me">
<c:url value="/servlet/DownloadServlet" var="url">
<c:param name="filename" value="${me.key}"></c:param>
</c:url>
${me.value}  <a href="${url}">下载</a><br/>
</c:forEach>
</body>

3.文件下载DownloadServlet:

public class DownloadServlet extends HttpServlet {

	public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=UTF-8");
OutputStream out = response.getOutputStream(); String filename = request.getParameter("filename");//get请求方式
filename = new String(filename.getBytes("ISO-8859-1"),"UTF-8");//中文编码 //截取老文件名称
String oldFileName = filename.split("_")[1];
//得到存储路径
String storePath = getServletContext().getRealPath("/WEB-INF/files");
//得到文件的所有路径
String filePath = makeStorePath(storePath, filename)+"\\"+filename; //推断文件是否存在
File file = new File(filePath);
if(!file.exists()){
out.write("对照起。你要下载的文件可能已经不存在了".getBytes("UTF-8"));
return;
} InputStream in = new FileInputStream(file);
//通知client下面载的方式打开
response.setHeader("Content-Disposition", "attachment;filename="+URLEncoder.encode(oldFileName, "UTF-8")); byte[] b = new byte[1024];
int len = -1;
while((len=in.read(b))!=-1){
out.write(b, 0, len);
}
in.close();
out.write("下载成功".getBytes("UTF-8"));
} public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException { doGet(request, response);
}
private String makeStorePath(String storePath, String fileName) {
int hashCode = fileName.hashCode();
int dir1 = hashCode & 0xf;// 0000~1111:整数0~15共16个
int dir2 = (hashCode & 0xf0) >> 4;// 0000~1111:整数0~15共16个 String path = storePath + "\\" + dir1 + "\\" + dir2; // WEB-INF/files/1/12
File file = new File(path);
if (!file.exists())
file.mkdirs(); return path;
}
}

JSP中文件的上传于下载演示样例的更多相关文章

  1. JavaWeb中文件的上传和下载

    JavaWeb中文件的上传和下载 转自: JavaWeb学习总结(五十)——文件上传和下载 - 孤傲苍狼 - 博客园https://www.cnblogs.com/xdp-gacl/p/4200090 ...

  2. iOS开发中文件的上传和下载功能的基本实现-备用

    感谢大神分享 这篇文章主要介绍了iOS开发中文件的上传和下载功能的基本实现,并且下载方面讲到了大文件的多线程断点下载,需要的朋友可以参考下 文件的上传 说明:文件上传使用的时POST请求,通常把要上传 ...

  3. Java中文件的上传与下载

    文件的上传与下载主要用到两种方法: 1.方法一:commons-fileupload.jar  commons-io.jar apache的commons-fileupload实现文件上传,下载 [u ...

  4. 009 spring boot中文件的上传与下载

    一:任务 1.任务 文件的上传 文件的下载 二:文件的上传 1.新建一个对象 FileInfo.java package com.cao.dto; public class FileInfo { pr ...

  5. Struts中文件的上传与下载

    前面学到的用组件去上传 前台: 1.post表单提交 2.表单类型 multipart/form-data 3.intput type=file 后台: Apach提供的FileUpload组件 核心 ...

  6. Struts2中文件的上传与下载

    文件上传 1.jsp页面 <s:form action="fileAction" namespace="/file" method="POST& ...

  7. docker容器中文件的上传与下载

    原文地址:传送门 1.上传文件 docker cp [OPTIONS] SRC_PATH|- CONTAINER:DEST_PATH [OPTIONS]:保持源目标中的链接,例: docker cp ...

  8. Jsp/Servlet文件的上传和下载

    文件上传的入门 文件上传的步骤:       总结实现思路: 1.创建核心上传类ServletFileUpload,这个时候需要一个工厂类 2.创建磁盘工厂类对象DiskFileItemFactory ...

  9. 文件的上传和下载--SpringMVC

    文件的上传和下载是项目开发中最常用的功能,例如图片的上传和下载.邮件附件的上传和下载等. 接下来,将对Spring MVC环境中文件的上传和下载进行详细的讲解. 一.文件上传 多数文件上传都是通过表单 ...

随机推荐

  1. Codefroces Educational Round 26 837 B. Flag of Berland

    B. Flag of Berland time limit per test 1 second memory limit per test 256 megabytes input standard i ...

  2. unalias---取消命令别名

    unalias命令用来取消命令别名,是为shell内建命令. 选项 -a:取消所有命令别名. 实例 使用unalias命令将已经设置的命令别名"cc"取消,输入如下命令: unal ...

  3. php汉字转化为拼音函数

    <?php function Pinyin($_String, $_Code='gb2312'){ $_DataKey = "a|ai|an|ang|ao|ba|bai|ban|ban ...

  4. JNDI学习总结(1)——JNDI入门

    JNDI是 Java 命名与目录接口(Java Naming and Directory Interface),在J2EE规范中是重要的规范之一,不少专家认为,没有透彻理解JNDI的意义和作用,就没有 ...

  5. 迅雷云监工crysadm搭建

    之前在淘宝众筹买了个迅雷赚钱宝Pro,2016年01月到手,玩了半个月. 之后在看百度贴吧时,看到好多大神都在用PC上用网页版软件来管理赚钱宝.非常是好奇.搜索了非常多资料,基本上都已经过时了.都无法 ...

  6. java初探秘之推断输入的一串字符是否全为小写字母

    import java.io.IOException; import java.util.*; public class Two { public static void main(String[] ...

  7. JavaScript作用域闭包(你不知道的JavaScript)

    JavaScript闭包.是JS开发project师必须深入了解的知识. 3月份自己曾撰写博客<JavaScript闭包>.博客中仅仅是简单阐述了闭包的工作过程和列举了几个演示样例,并没有 ...

  8. jquery js解析函数、函数直接调用

    ----------------------------------------------------------------- cc = function(){alert(345)}, pushS ...

  9. cluster discovery概述及FaultDetection分析

    elasticsearch cluster实现了自己发现机制zen.Discovery功能主要包括以下几部分内容:master选举,master错误探测,集群中其它节点探测,单播多播ping.本篇会首 ...

  10. if 的理解

    1. if 实现集合的划分 比如著名的 Prim 算法(最小生成树),从某一确定的点出发,每次新加入的点,都是在已访问过的结点(u∈U)和未访问过(v∈V−U)的结点之间的边.这里的未被访问(V−U) ...