使用JSP做文件上传

14 篇文章 0 订阅

1 文件上传表单

  1)上传文件的本质是文本复制的过程
  2)技术层面,在Java中一定会用到IO操作,主要以二进制方式读写
  3)传统方式下,对于上传文件字段不同的浏览器有着不同的解析方式,例如:
IE6:
upfile=c:\aa\bb\a.JPG
非IE6: 
upfile=a.JPG
  4)可以将form以MIME协议的方式将上传文件传递到服务端,服务端以二进制流的方式读写
代码: 客户端form enctype="multipart/form-data"/>
      服务端request.getInputStream()


*2 上传文件的细节

  (1)中文乱码的问题
     a)普通字段的中文乱码问题:FileItem.getString("UTF-8"),
       注意:FileItem表示上传表单中的表单项内容
     b)上传字段的中文乱码问题:ServletUploadFile.setHeaderEncoding("UTF-8");


  (2)临时文件的删除的问题
     a)通过FileItem.delete()
     b)一定要在关闭IO流之后


  (3)在同一个目录下上传相同文件名的问题
     a)将文件名拼接一个唯一标识符,即UUID


  (4)单个目录下文件过多的问题
     a)采用位运算解决单个目录文件过多
     b)思路:参见<<>>


  (5)为安全将上传的文件放入客户端无法直接访问的目录中的问题
     a)将上传的文件,放置到/WEB-INF/upload/目录下
     
  (6)重构思想
     a)做到通用性


  (7)自定义封装上传文件工具类的问题


课堂练习:无上传文件提交/只能上传JPG或jpg文件


  (8)上传文件的大小的问题


  (9)上传多个文件的问题


 (10)上传多个文件的界面问题


课堂练习:基于界面,限制上传文件的个数,参见<<课堂练习1.JPG>>

*3 显示下载文件列表

  a)递归方式查询可供下载的文件,一定要有出口条件
  b)使用Map<UUID文件名,真实文件名>收集可供下载的文件
  c)使用<c:url>和<c:param>对中文名进行URL编码

*4 下载文件

   a)对传过来的中文编码进行URL解码
   b)通过UUID文件名,反向查到该文件所在的真实目录

  

 5 文件上传下载与数据库结合

   a)在将上传文件保存的同时,写往数据库表,一个上传文件对应一条记录,确保uuidFileName双方一致
   
   思考:
a)上传时,先硬盘,再表?
b)下载时,先硬盘,再表?
c)删除时,先硬盘,再表?

d)是否需要事务支持?

<html>
  <head>
  	<script type="text/javascript">
  		//全局变量
  		var time = 0;
  		function addLine(addButton){
  			//创建内部div对象
  			var divElement = document.createElement("div");
  			//创建input对象[file类型]
  			var inputElement1 = document.createElement("input");
  			inputElement1.type="file";
  			inputElement1.name="upfile";
  			//创建input对象[button类型]
  			var inputElement2 = document.createElement("input");
  			inputElement2.type="button";
  			inputElement2.value="删除";
  			//对删除按钮添加事件监听
  			inputElement2.οnclick=function(){
  				//取得该按钮所在行的直接父元素
  				var divElement = this.parentNode.parentNode;
  				//通过父元素删除直接子元素
  				divElement.removeChild(this.parentNode);
  				time--;
  				if(time < 5){
  					//按钮生效
  					addButton.disabled=false;
  					//addButton.style.visibility="visible";
  				}
  			}
  			//依次将file类型和button类型的input对象加入到内部div对象中
  			divElement.appendChild(inputElement1);
  			divElement.appendChild(inputElement2);
  			//再次内部div对象加入到外部div对象
  			var outDivElement = document.getElementById("outDiv");
  			outDivElement.appendChild(divElement);
  			time++;
  			if(time == 5){
  				//将按钮失效
  				addButton.disabled=true;
  				//addButton.style.visibility="hidden";
  			}	
  		}
  	</script>
  </head>
  <body>
  	<form 
  		action="${pageContext.request.contextPath}/UploadServlet" 
  		method="POST"
  		enctype="multipart/form-data">
  		<table border="1" align="center">
  			<caption>文件上传</caption>
  			<tr>
  				<th>上传用户</th>
  				<td><input type="text" name="username"/></td>
  			</tr>
  			<tr>
  				<th></th>
  				<td>
  					<div id="outDiv">
  						<%--
  						<div>
							<input type="file" name="upfile"/>
							<input type="button" value="删除"/>  							
  						</div>
  						--%>
  					</div>
  				</td>
  			</tr>
  			<tr>
  				<th></th>
  				<td>
  					<input 
  						type="button" 
  						value="添加上传文件"
  						οnclick="addLine(this)"
  					/>
  				</td>
  			</tr>
  			<tr>
  				<td colspan="2" align="center">
  					<input type="submit" value="上传"/>
  					<a href="${pageContext.request.contextPath}/ListFileServlet">
  						显示下载文件
  					</a>
  				</td>
  			</tr>
  		</table>
  	</form>
  </body>
</html>
写Servlet控制上传文件:
public class UploadServlet extends HttpServlet {
	public void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {
		try {
			User user = UploadUtil.doUpload(request);
			String uploadPath = this.getServletContext().getRealPath(UploadUtil.uploadPath);
			
			List<Up> upList = new ArrayList<Up>();
			
			//写入硬盘
			UploadUtil.doSave(user,uploadPath,upList);
			
			//写入数据库表
			UpService upService = new UpService();
			upService.addUps(upList);
			
			request.setAttribute("message","上传文件成功");
			request.getRequestDispatcher("/WEB-INF/message.jsp").forward(request,response);
		}catch(UpfileSizeException e){
			e.printStackTrace();
			request.setAttribute("message","<font color='green'>上传文件大小限制在200K以内</font>");
			request.getRequestDispatcher("/WEB-INF/message.jsp").forward(request,response);
		}catch(UpfileTypeException e){
			e.printStackTrace();
			request.setAttribute("message","<font color='red'>只能上传JPG格式的文件</font>");
			request.getRequestDispatcher("/WEB-INF/message.jsp").forward(request,response);
		}catch(NoUpfileException e){
			e.printStackTrace();
			request.setAttribute("message","<font color='blue'>无上传文件</font>");
			request.getRequestDispatcher("/WEB-INF/message.jsp").forward(request,response);
		}catch (Exception e) {
			e.printStackTrace();
			request.setAttribute("message","上传文件失败");
			request.getRequestDispatcher("/WEB-INF/message.jsp").forward(request,response);
		}
	}
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
本代码主要应用的是jsp技术,而实现的文件上传功能,这个功能也是比较常见的,也是比较常用的,更是在网络中比较流行的。 技术为创建显示动态生成内容的Web页面提供了一个简捷而快速的方法。JSP技术的设计目的是使得构造基于Web的应用程序更加容易和快捷,而这些应用程序能够与各种Web服务器,应用服务器,浏览器和开发工具共同工作。 Web应用开发的JavaServer Pages技术方法 在开发JSP规范的过程中,太阳微系统公司(Sun Microsystems Inc.)与许许多多主要的Web服务器、应用服务器和开发工具供应商,以及各种各样富有经验的开发团体进行合作。其结果是找到了一种为应用和页面开发人员平衡了可移植性和易用性的开发方法。 JSP技术在多个方面加速了动态Web页面的开发: 将内容的生成和显示进行分离 使用JSP技术,Web页面开发人员可以使用HTML或者XML标识来设计和格式化最终页面。使用JSP标识或者小脚本来生成页面上的动态内容(内容是根据请求来变化的,例如请求帐户信息或者特定的一瓶酒的价格)。生成内容的逻辑被封装在标识和JavaBeans组件中,并且捆绑在小脚本中,所有的脚本在服务器端运行。如果核心逻辑被封装在标识和Beans中,那么其他人,如Web管理人员和页面设计者,能够编辑和使用JSP页面,而不影响内容的生成。 在服务器端,JSP引擎解释JSP标识和小脚本,生成所请求的内容(例如,通过访问JavaBeans组件,使用JDBCTM技术访问数据库,或者包含文件),并且将结果以HTML(或者XML)页面的形式发送回浏览器。这有助于作者保护自己的代码,而又保证任何基于HTML的Web浏览器的完全可用性。 强调可重用的组件 绝大多数JSP页面依赖于可重用的,跨平台的组件(JavaBeans或者Enterprise JavaBeansTM组件)来执行应用程序所要求的更为复杂的处理。开发人员能够共享和交换执行普通操作的组件,或者使得这些组件为更多的使用者或者客户团体所使用。基于组件的方法加速了总体开发过程,并且使得各种组织在他们现有的技能和优化结果的开发努力中得到平衡。 采用标识简化页面开发 Web页面开发人员不会都是熟悉脚本语言的编程人员。JavaServer Page技术封装了许多功能,这些功能是在易用的、与JSP相关的XML标识中进行动态内容生成所需要的。标准的JSP标识能够访问和实例化JavaBeans组件,设置或者检索组件属性,下载Applet,以及执行用其他方法更难于编码和耗时的功能。 通过开发定制化标识库,JSP技术是可以扩展的。今后,第三方开发人员和其他人员可以为常用功能创建自己的标识库。这使得Web页面开发人员能够使用熟悉的工具和如同标识一样的执行特定功能的构件来工作。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值