servlet 上传

上传 源码day22_1

客户端给服务端发送请求相当于是客户端给服务端发送字符串,但是上传可以理解为,客户端给服务端发送文件

 

1. 上传对表单限制

  * 表单中需要添加文件表单项:<input type="file" name="xxx" />(name 是 必须要写的)
  * method="post" (所以文件的内容不能列在url后面,请求必须有体)
  * enctype="multipart/form-data"(多部件的表单数据,有点类似于url 编码),这句话到导致request.getParametere("xxx");这个方法在表单为enctype="multipart/form-data"时,它作废了,所以在调用的时候,不能知道他调用的是什么方法。

多部件表单的体

1. 每隔出多个部件,即一个表单项一个部件。
2. 一个部件中自己包含请求头和空行,以及请求体。
3. 普通表单项:
  > 1个头:Content-Disposition:包含name="xxxx",即表单项名称。
  > 体就是表单项的值
4. 文件表单项:
  > 2个头:
    * Content-Disposition:包含name="xxxx",即表单项名称;还有一个filename="xxx",表示上传文件的名称
    * Content-Type:它是上传文件的MIME类型,例如:image/pjpeg,表示上传的是图片,图上中jpg扩展名的图片。
  > 体就是上传文件的内容。

  <body>
<h1>上传1</h1>
<form action="<c:url value='/Upload1Servlet'/>" method="post" enctype="multipart/form-data">
  用户名;<input type="text" name="username"/><br/>
  照 片:<input type="file" name="zhaoPian"/><br/>
  <input type="submit" value="上传"/>
</form>

 

package cn.itcast.servlet;

import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.commons.io.IOUtils;

public class Upload1Servlet extends HttpServlet {
	public void doPost(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {
		request.setCharacterEncoding("utf-8");
		response.setContentType("text/html;charset=utf-8");
		
		ServletInputStream request.getInputStream();//包含整个请求的体!
		String s = IOUtils.toString(in);
		System.out.println(s);
	}
}

 上传的时候只能是识别jpg 格式的,当点击上传图片的按钮时返回来就是io流当中的内容

下面就是io 流打印出来的内容

这个里面就是多部件表单的内容,但是下面看起来是很多部分,阅读起来很不容易,所以采用的是

commons-fileupload
  * commons-fileupload.jar 必须依赖commons-io.jar
  * commons-io.jar 这个jar 包可以自己独立运行

commons-fileupload.jar 这个小组件,它会帮我们解析request中的上传数据,解析后的结果是一个表单项数据封装到一个FileItem对象中。我们只需要调用FileItem的方法即可!

2. 上传对Servlet限制
  * request.getParametere("xxx");这个方法在表单为enctype="multipart/form-data"时,它作废了。它永远都返回null,由于在代码中没有这句话了,那么request.setCharacterEncoding("utf-8"); 这句话在代码中不就没有用了吗,但是commons-fileupload.jar  依然会用到request.setCharacterEncoding("utf-8"); 这句话
  * ServletInputStream request.getInputStream();包含整个请求的体!

1. 上传三步
  相关类:
  * 工厂:DiskFileItemFactory
  * 解析器:ServletFileUpload


  * 表单项:FileItem

  1). 创建工厂:DiskFileItemFactory factory = new DiskFileItemFactory();


  2). 创建解析器:ServletFileUpload sfu = new ServletFileUpload(factory);

需要查看的是commons-fileupload.jar 的api


  3). 使用解析器来解析request,得到FileItem集合:List<FileItem> fileItemList = sfu.parseRequest(request);

2. FileItem
  * boolean isFormField():是否为普通表单项!返回true为普通表单项,如果为false即文件表单项!
  * String getFieldName():返回当前表单项的名称;
  * String getString(String charset):返回表单项的值;
  * String getName():返回上传的文件名称
  * long getSize():返回上传文件的字节数
  * InputStream getInputStream():返回上传文件对应的输入流(其实就是相当于得到了上传文件是一样的,可以写到任何地方去)
  * void write(File destFile):把上传的文件内容保存到指定的文件中,如果这个文件存在的话,那么会覆盖这个文件,如果没有这个文件的话,会创建这个文件
  * String getContentType();

package cn.itcast.servlet;

import java.io.File;
import java.io.IOException;
import java.util.List;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;

public class Upload2Servlet extends HttpServlet {
	public void doPost(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {
		request.setCharacterEncoding("utf-8");
		response.setContentType("text/html;charset=utf-8");
		
		/*
		 * 上传三步
		 * 1. 得到工厂
		 * 2. 通过工厂创建解析器
		 * 3. 解析request,得到FileItem集合
		 * 4. 遍历FileItem集合,调用其API完成文件的保存
		 */
		DiskFileItemFactory factory = new DiskFileItemFactory();
		ServletFileUpload sfu = new ServletFileUpload(factory);
		try {
			List<FileItem> fileItemList = sfu.parseRequest(request);
			FileItem fi1 = fileItemList.get(0);//得到的是普通表单项
			FileItem fi2 = fileItemList.get(1);//得到的是文件表单项
			
			System.out.println("普通表单项演示:" + fi1.getFieldName() + 
					"=" + fi1.getString("UTF-8"));
			System.out.println("文件表单项演示:");
			System.out.println("Content-Type: " + fi2.getContentType());
			System.out.println("size: " + fi2.getSize());
			System.out.println("filename: " + fi2.getName());
			
			// 保存文件
			File destFile = new File("c:/baibing.jpg");
			fi2.write(destFile);
		} catch (FileUploadException e) {
			throw new RuntimeException(e);
		} catch (Exception e) {
			throw new RuntimeException(e);
		}
	}
}

  </head>
  
  <body>
<h1>上传2</h1>
<form action="<c:url value='/Upload2Servlet'/>" method="post" enctype="multipart/form-data">
  用户名;<input type="text" name="username"/><br/>
  照 片:<input type="file" name="zhaoPian"/><br/>
  <input type="submit" value="上传"/>
</form>
  </body>
</html>

后台打印出来的

INFO: Server startup in 2293 ms
七月 20, 2018 11:56:30 上午 org.apache.jasper.compiler.TldLocationsCache tldScanJar
INFO: At least one JAR was scanned for TLDs yet contained no TLDs. Enable debug logging for this logger for a complete list of JARs that were scanned but no TLDs were found in them. Skipping unneeded JARs during scanning can improve startup time and JSP compilation time.
普通表单项演示:username=admin
文件表单项演示:
Content-Type: image/jpeg
size: 7577
filename: 100344.jpg

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

上传的细节:

1. 文件必须保存到WEB-INF下!也可以保存在其他硬盘中
  * 目的是不让浏览器直接访问到!
  * 把文件保存到WEB-INF目录下!

但是如果保存的文件疏忽了没有保存到WEB-INFO 下面 而是保存到

而是保存上面的文件路径下了,项目中,如果上传的是index.jsp 的话就很危险了

例如保存一个index.jsp jsp 当中写这么一段代码

由于在runtime 方法中可以执行一些dos 命令

例如


2. 文件名称相关问题
     1) 有的浏览器上传的文件名是绝对路径,这需要切割!C:\files\baibing.jpg(ie6 可能会出现这种情况)
            String filename = fi2.getName();
              int index = filename.lastIndexOf("\\"); "\\" 表示的是一个斜杠的意思
              if(index != -1) {
              filename = filename.substring(index+1); 要从b开始截取
              }
       2)文件名乱码或者普通表单项乱码:

文件名没有乱码的原因是因为   request.setCharacterEncoding("utf-8");  这句话,那么还有没有其他的方式也可以防止乱码呢

当上传的谁的名称中包含中文时,需要设置编码,commons-fileupload组件为我们提供了两种设置编码的方式:

  1. request.setCharacterEncoding(String):这种方式是我们最为熟悉的方式了,不是commons-fileupload 带的原来就有的
  2. fileUpload.setHeaderEncdoing(String):如果两种方式同时使用,这种方式的优先级高与前一种,这种方式是commons-fileupload 带的

上传文件的文件内容包含中文:

通常我们不需关心上传文件的内容,因为我们会把上传文件保存到硬盘上!也就是说,文件原来是什么样子,到服务器这边还是什么样子!

但是如果你有这样的需求,非要在控制台显示上传的文件内容,那么你可以使用fileItem.getString(“utf-8”)来处理编码。

文本文件内容和普通表单项内容使用FileItem类的getString(“utf-8”)来处理编码。

      3) 文件同名问题;我们需要为每个文件添加名称前缀,这个前缀要保证不能重复。uuid
    > filename = CommonUtils.uuid() + "_" + filename; 这就是因为我们从网上下载的时候文件的名字是很乱的。

 
public class CommonUtils {
	public static String uuid() {
		return UUID.randomUUID().toString().replace("-", "").toUpperCase();
	}
}

3      一个目录不能存放过多的文件(存放目录打散)

        一个目录下不应该存放过多的文件,一般一个目录存放1000个文件就是上限了,如果在多,那么打开目录时就会很“卡”。你  可以尝试打印C:\WINDOWS\system32目录,你会感觉到的。

也就是说,我们需要把上传的文件放到不同的目录中。但是也不能为每个上传的文件一个目录,这种方式会导致目录过多。所以我们应该采用某种算法来“打散”!

打散的方法有很多,例如使用日期来打散,每天生成一个目录。也可以使用文件名的首字母来生成目录,相同首字母的文件放到  同一目录下。

打散方式:

     1)日期打散算法:使用当前日期做为目录(如果某一天上传的文件过多,那么也会出现一个目录文件过多的情况);

     2) 首字符打散算法:如果文件名是中文的,因为中文过多,所以会导致目录过多的现象。使用文件的首字母做为目录名称,例如:abc.txt,那么我们把文件保存到a目录下。如果a目录这时不存在,那么创建之。
    > 哈希打散:任何对象都有一个方法是hashcode 方法


      * 通过文件名称得到int值,即调用hashCode()

        例如 任何字符串都是对象 "fdafdfa".hashCode(); 
      * 它int值转换成16进制0~9, A~F
      * 获取16进制的前两位用来生成目录,目录为二层!例如:文件名的hashcode()的值转化为16进制为1B2C3D4E5F,只要前两位,例如是1B , 第一层最多是16个,第二层最多也是16个,那么一共就是256个

例子:

package cn.itcast.servlet;

import java.io.File;
import java.io.IOException;
import java.util.List;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadBase;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;

import cn.itcast.commons.CommonUtils;

public class Upload3Servlet extends HttpServlet {

	public void doPost(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {
		request.setCharacterEncoding("utf-8");
		response.setContentType("text/html;charset=utf-8");
		//上传三步
		// 工厂
		DiskFileItemFactory factory = new DiskFileItemFactory();
		// 解析器
		ServletFileUpload sfu = new ServletFileUpload(factory);
//		sfu.setFileSizeMax(100 * 1024);//限制单个文件大小为100K
//		sfu.setSizeMax(1024 * 1024);//限制整个表单大小为1M
		
		// 解析,得到List
		try {
			List<FileItem> list = sfu.parseRequest(request);
			FileItem fi = list.get(1);
			//1. 得到文件保存的路径
			String root = this.getServletContext().getRealPath("/WEB-INF/files/");//是在这个目录下打散并生成两级目录
			System.out.println("得到文件保存的路径"+root);
//			得到文件保存的路径E:\workspaceforeclipse\.metadata\.plugins\org.eclipse.wst.server.core\tmp1\wtpwebapps\day22_1\WEB-INF\files
			/*
			 * 2. 生成二层目录
			 *   1). 得到文件名称
			 *   2). 得到hashCode
			 *   3). 转发成16进制
			 *   4). 获取前二个字符用来生成目录
			 */
			String filename = fi.getName();//获取上传的文件名称
			 //处理文件名的绝对路径问题
			int index = filename.lastIndexOf("\\");
			if(index != -1) {
				filename = filename.substring(index+1);
			}
			//给文件名称添加uuid前缀,处理文件同名问题
			String savename = CommonUtils.uuid() + "_" + filename;
			
			//1. 得到hashCode
			int hCode = filename.hashCode();
			String hex = Integer.toHexString(hCode);//把hashcode 转化为16进制
			//2. 获取hex的前两个字母,与root连接在一起,生成一个完整的路径
			File dirFile = new File(root, hex.charAt(0) + "/" + hex.charAt(1));
			System.out.println(dirFile.toString());
//			E:\workspaceforeclipse\.metadata\.plugins\org.eclipse.wst.server.core\tmp1\wtpwebapps\day22_1\WEB-INF\files\e\e
			//3. 创建目录链
			dirFile.mkdirs();
			 //4. 创建目录文件
			File destFile = new File(dirFile, savename);
//			File destFile = new File(File parent, String child);
			//5. 保存
			fi.write(destFile);
			
		} catch (FileUploadException e) {
			if(e instanceof FileUploadBase.FileSizeLimitExceededException) {
				request.setAttribute("msg", "您上传的文件超出了100KB!");
				request.getRequestDispatcher("/form3.jsp").forward(request, response);
			}
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

  


4. 上传文件的大小限制
  1) 单个文件大小限制
    > sfu.setFileSizeMax(100*1024):限制单个文件大小为100KB
    > 上面的方法调用,必须在解析开始之前调用!


   

> 如果上传的文件超出限制,在parseRequest()方法执行时,会抛出异常!FileUploadBase.FileSizeLimitExceededException

  2) 整个请求所有数据大小限制(因为可能有多个文件上传)
    > sfu.setSizeMax(1024 * 1024);//限制整个表单大小为1M
    > 这个方法也是必须在parseRequest()方法之前调用
    > 如果上传的文件超出限制,在parseRequest()方法执行时,会抛出异常!FileUploadBase.SizeLimitExceededException

抛出的异常在后面

对单独文件超出100kb 进行测试

package cn.itcast.servlet;

import java.io.File;
import java.io.IOException;
import java.util.List;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadBase;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;

import cn.itcast.commons.CommonUtils;

public class Upload3Servlet extends HttpServlet {

	public void doPost(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {
		request.setCharacterEncoding("utf-8");
		response.setContentType("text/html;charset=utf-8");
		//上传三步
		// 工厂
		DiskFileItemFactory factory = new DiskFileItemFactory();
		// 解析器
		ServletFileUpload sfu = new ServletFileUpload(factory);
		sfu.setFileSizeMax(100 * 1024);//限制单个文件大小为100K  这个调用必须在解析开始之前,要不就不会生效的
		sfu.setSizeMax(1024 * 1024);//限制整个表单大小为1M   这个调用必须在解析开始之前,要不就不会生效的
		
		// 解析,得到List
		try {
			List<FileItem> list = sfu.parseRequest(request);
			FileItem fi = list.get(1);
			//1. 得到文件保存的路径
			String root = this.getServletContext().getRealPath("/WEB-INF/files/");//是在这个目录下打散并生成两级目录
			System.out.println("得到文件保存的路径"+root);
//			得到文件保存的路径E:\workspaceforeclipse\.metadata\.plugins\org.eclipse.wst.server.core\tmp1\wtpwebapps\day22_1\WEB-INF\files
			/*
			 * 2. 生成二层目录
			 *   1). 得到文件名称
			 *   2). 得到hashCode
			 *   3). 转发成16进制
			 *   4). 获取前二个字符用来生成目录
			 */
			String filename = fi.getName();//获取上传的文件名称
			 //处理文件名的绝对路径问题
			int index = filename.lastIndexOf("\\");
			if(index != -1) {
				filename = filename.substring(index+1);
			}
			//给文件名称添加uuid前缀,处理文件同名问题
			String savename = CommonUtils.uuid() + "_" + filename;
			
			//1. 得到hashCode
			int hCode = filename.hashCode();
			String hex = Integer.toHexString(hCode);//把hashcode 转化为16进制
			//2. 获取hex的前两个字母,与root连接在一起,生成一个完整的路径
			File dirFile = new File(root, hex.charAt(0) + "/" + hex.charAt(1));
			System.out.println(dirFile.toString());
//			E:\workspaceforeclipse\.metadata\.plugins\org.eclipse.wst.server.core\tmp1\wtpwebapps\day22_1\WEB-INF\files\e\e
			//3. 创建目录链
			dirFile.mkdirs();
			 //4. 创建目录文件
			File destFile = new File(dirFile, savename);
//			File destFile = new File(File parent, String child);
			//5. 保存
			fi.write(destFile);
			
		} catch (FileUploadException e) {
			if(e instanceof FileUploadBase.FileSizeLimitExceededException) {
				request.setAttribute("msg", "您上传的文件超出了100KB!");
				request.getRequestDispatcher("/form3.jsp").forward(request, response);
			}
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}
<h1>上传3</h1>
<h3>${msg }</h3>
<form action="<c:url value='/Upload3Servlet'/>" method="post" enctype="multipart/form-data">
  用户名;<input type="text" name="username"/><br/>
  照 片:<input type="file" name="zhaoPian"/><br/>
  <input type="submit" value="上传"/>
</form>
  </body>
</html>

 

 

 

 


 


5. 缓存大小与临时目录
  * 缓存大小:超出多大,才向硬盘保存!默认为10KB
  * 临时目录:向硬盘的什么目录保存
  设置缓存大小与临时目录:new DiskFileItemFactory(20*1024, new File("F:/temp"))

//创建20k的缓存的大小只有上传的文件超过20k 才会存储在存储在临时文件中
//DiskFileItemFactory(int sizeThreshold, java.io.File repository) 第一个参数是缓存的大小,第二个参数是临时文件放在的目录
DiskFileItemFactory factory = new DiskFileItemFactory(20*1024, new File("e:/temp"));//"e:/temp" 这个文件的目录是必须要存在的
// 解析器
ServletFileUpload sfu = new ServletFileUpload(factory);
//如果采用临时目录,就不用限制上传文件的大小了
sfu.setFileSizeMax(100 * 1024);//限制单个文件大小为100K  这个调用必须在解析开始之前,要不就不会生效的  
//如果采用临时目录,就不用限制上传文件的大小了
sfu.setSizeMax(1024 * 1024);//限制整个表单大小为1M   这个调用必须在解析开始之前,要不就不会生效的

临时文件是当上传成功了以后才会消失

上传成功以后文件夹就空了

就上传到WEB-INFO 下面的files中的

E:\workspaceforeclipse\.metadata\.plugins\org.eclipse.wst.server.core\tmp1\wtpwebapps\day22_1\WEB-INF\files\d\7

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值