- 浏览: 1208635 次
- 性别:
- 来自: 上海
-
文章分类
- 全部博客 (361)
- java综合 (33)
- 项目管理 (10)
- 工作流 (6)
- spring (11)
- hibenate (17)
- struts (0)
- javascript,html,css,ajax,jquery (11)
- IDE (9)
- 生活 (0)
- 工作 (0)
- 杂记 (1)
- 数据库 (96)
- 服务器 (5)
- 可视编辑 (0)
- freemarker (6)
- 操作系统 windows (13)
- web页面 (6)
- jms (15)
- 调优 (4)
- 测试和bug管理 (2)
- 原理 (1)
- 項目-atf (17)
- 安全 (3)
- xml (4)
- 操作系统 liunx (21)
- 网络 (22)
- office (11)
- 设计 (2)
- 软件 (1)
- 数据库 mysql (6)
- 胖客户端-flex (1)
- 正则 (9)
- oracle- liunx (3)
- sql2000 (2)
- 模式 (1)
- 虚拟机 (2)
- jstl (2)
- 版本控制 打包工具 (0)
- AOP (1)
- demo (1)
- 小软件 (2)
- 感恩 (1)
- iphone 4 (1)
- 反欺诈业务经验整理 (0)
最新评论
-
sea0108:
mark
java内存模型 -
XingShiYiShi:
方便把:testPNR();具体实现发出来吗?谢谢
用正则表达式解析 航信的电子客票和pnr报文 -
wh359126613:
如果js和webservice不在同一个服务器上,有跨域问题如 ...
使用javascript调用webservice示例 -
雨飛雁舞:
...
oracle 动态性能(V$)视图 -
ouyang1224:
好东西
oracle 动态性能(V$)视图
掌控上传进度的AJAX Upload
作者:cleverpig
AJAX——最酷的“冲浪板”
原文永久链接及源代码下载地址:
http://www.matrix.org.cn/resource/article/2007-01-08/09db6d69-9ec6-11db-ab77-2bbe780ebfbf.html
动机:
2006年底Google了一下AJAX Upload实现,结果没有发现很完整的Java实现。硕果仅存的就是TELIO公司的Pierre-Alexandre发表的《AJAX Upload progress monitor for Commons-FileUpload Example》文中提供的ajax-upload-1.0.war。
虽然上文中完成Upload工作的是Apache的Common-FileUpload组件,但在其代码中所使用的FileUpload1.1版本并没有1.2版本所提供的上传处理Listener功能,这就对检测文件上传情况造成了困难。我想正是这个原因致使Pierre-Alexandre使用了DWR+MonitoredDiskFileItem、MonitoredDiskFileItemFactory类(分别继承DiskFileItem、DiskFileItemFactory) 的方式:前者负责在web客户端进行Remote Call;后者在进行文件数据读取时统计数据总量、读取数据量、处理文件总数,并保存于Session中,以供web客户端通过DWR远程调用 UploadMonitor类的getUploadInfo方法进行轮询(Poll)。
从本人观点出发,Pierre-Alexandre实现的不足之处:
1.没有用户取消上传功能;
2.完全的DWR实现,没有使用Prototype,对于不会使用DWR的开发者来讲有一定的知识局限性,而且由于DWR的个性而造成不便将此实现集成到项目中。
Prototype+Servlet的实现:
Prototype+Servlet的Example
所以出于研究Prototype之目的,本人经过仔细思考,尝试实现了一个Prototype+Servlet的简单Example。其工作流程很简单:
1.在Form提交上传文件Field的同时,使用AJAX周期性地从Servlet轮询上传状态信息;
2.然后,根据此信息更新进度条和相关文字,及时反映文件传输状态;
3.如果用户取消上传操作,则进行相应的现场清理工作:删除已经上传的文件,在Form提交页面中显示相关信息;
4.如果上传完毕,在Form提交页面中显示已经上传的文件内容(或链接),也可以与一些AJAX SlideShow应用结合在一起。
服务器端代码:
Bean序列化/反序列化工作:XmlUnSerializer这个类虽然不能够通吃任何模样的Bean,但应付一般的Bean、具有Collection类型属性的Bean和Bean List来讲还是够用的。
{XmlUnSerializer类的核心方法serializeBean和serializeBeanList}:
java 代码
- /**
- * 将bean系列化为UTF-8编码的xml
- * @param beanObj
- * @return
- * @throws IOException
- */
- public static String serializeBean(Object beanObj) throws IOException{
- …
- }
- /**
- * 将bean列表序列化为UTF-8编码的xml
- * @param beanObj
- * @return
- * @throws IOException
- */
- public static String serializeBeanList(Object beanListObj) throws IOException{
- …
- }
文件上传状态Bean:使 用FileUploadStatus这个类记录文件上传状态,并将其作为服务器端与web客户端之间通信的媒介物:通过对这个类对象进行XML序列化作为 服务器回应发送给web客户端,web客户端使用JavaScript对其进行反序列化处理获得JavaScript版本的文件上传状态对象。
{FileUploadStatus的属性}:
java 代码
- //上传总量
- private long uploadTotalSize=0;
- //读取上传总量
- private long readTotalSize=0;
- //当前上传文件号
- private int currentUploadFileNum=0;
- //成功读取上传文件数
- private int successUploadFileCount=0;
- //状态
- private String status="";
- //处理起始时间
- private long processStartTime=0l;
- //处理终止时间
- private long processEndTime=0l;
- //处理执行时间
- private long processRunningTime=0l;
- //上传文件URL列表
- private List uploadFileUrlList=new ArrayList();
- //取消上传
- private boolean cancel=false;
- //上传base目录
- private String baseDir="";
文件上传状态监视工作:使用Common-FileUpload 1.2版本(20070103)。此版本与1.1版的区别在于提供了能够监视文件上传情况的ProcessListener接口,使开发者通过FileUploadBase类对象的setProcessListener方法植入自己的Listener,而且实现这个Listener很简单。
{FileUploadListener主要方法update}:
java 代码
- /**
- * 更新状态
- * @param pBytesRead 读取字节总数
- * @param pContentLength 数据总长度
- * @param pItems 当前正在被读取的field号
- */
- public void update(long pBytesRead, long pContentLength, int pItems){
- FileUploadStatus fuploadStatus=BackGroundService.takeOutFileUploadStatusBean(this.session);
- logger.debug("当前正在处理第" + pItems+"个文件");
- fuploadStatus.setUploadTotalSize(pContentLength);
- //读取完成
- if (pContentLength == -1) {
- logger.debug("读取完成:读取了 " + pBytesRead + " bytes.");
- fuploadStatus.setStatus("完成对" + pItems+"个文件的读取:读取了 " + pBytesRead + " bytes.");
- fuploadStatus.setReadTotalSize(pBytesRead);
- fuploadStatus.setSuccessUploadFileCount(pItems);
- fuploadStatus.setProcessEndTime(System.currentTimeMillis());
- fuploadStatus.setProcessRunningTime(fuploadStatus.getProcessEndTime());
- //读取中
- } else {
- logger.debug("读取进行中:已经读取了 " + pBytesRead + " / " + pContentLength+ " bytes.");
- fuploadStatus.setStatus("当前正在处理第" + pItems+"个文件:已经读取了 " + pBytesRead + " / " + pContentLength+ " bytes.");
- fuploadStatus.setReadTotalSize(pBytesRead);
- fuploadStatus.setCurrentUploadFileNum(pItems);
- fuploadStatus.setProcessRunningTime(System.currentTimeMillis());
- }
- BackGroundService.storeFileUploadStatusBean(this.session,fuploadStatus);
- }
很清楚,我也把FileUploadStatus这个Bean存取于Session中。
Servlet实现:BackGroundService 这个Servlet类负责接收Form Post数据、回应状态轮询请求、处理取消文件上传的请求。尽管可以把这些功能相互分离开来(比如构造一个FileUploadManager类),但出 于简单明了、便于阅读之目的,还是将它们放到Servlet中,只是由不同的方法进行分割。
{BackGroundService中的processFileUpload方法用于处理文件上传请求}:
java 代码
- /**
- * 处理文件上传
- * @param request
- * @param response
- * @throws IOException
- * @throws ServletException
- */
- private void processFileUpload(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException{
- DiskFileItemFactory factory = new DiskFileItemFactory();
- //设置内存阀值,超过后写入临时文件
- factory.setSizeThreshold(10240000);
- //设置临时文件存储位置
- factory.setRepository(new File(request.getRealPath("/upload/temp")));
- ServletFileUpload upload = new ServletFileUpload(factory);
- //设置单个文件的最大上传size
- upload.setFileSizeMax(10240000);
- //设置整个request的最大size
- upload.setSizeMax(10240000);
- upload.setProgressListener(new FileUploadListener(request.getSession()));
- //保存初始化后的FileUploadStatus Bean
- storeFileUploadStatusBean(request.getSession(),initFileUploadStatusBean(request));
- String forwardURL="";
- try {
- List items = upload.parseRequest(request);
- //获得返回url
- for(int i=0;i
- FileItem item=(FileItem)items.get(i);
- if (item.isFormField()){
- logger.debug("form Field["+item.getFieldName()+"]="+item.getString());
- forwardURL=item.getString();
- break;
- }
- }
- //处理文件上传
- for(int i=0;i
- FileItem item=(FileItem)items.get(i);
- //取消上传
- if (takeOutFileUploadStatusBean(request.getSession()).getCancel()){
- deleteUploadedFile(request);
- break;
- }
- //保存文件
- else if (!item.isFormField() && item.getName().length()>0){
- String fileName=takeOutFileName(item.getName());
- logger.debug("处理文件["+fileName+"]:保存路径为"
- +request.getRealPath(UPLOAD_DIR)+File.separator+fileName);
- File uploadedFile = new File(request.getRealPath(UPLOAD_DIR)+File.separator+fileName);
- item.write(uploadedFile);
- //更新上传文件列表
- FileUploadStatus fUploadStatus=takeOutFileUploadStatusBean(request.getSession());
- fUploadStatus.getUploadFileUrlList().add(fileName);
- storeFileUploadStatusBean(request.getSession(),fUploadStatus);
- Thread.sleep(500);
- }
- }
- } catch (FileUploadException e) {
- logger.error("上传文件时发生错误:"+e.getMessage());
- e.printStackTrace();
- uploadExceptionHandle(request,"上传文件时发生错误:"+e.getMessage());
- } catch (Exception e) {
- // TODO Auto-generated catch block
- logger.error("保存上传文件时发生错误:"+e.getMessage());
- e.printStackTrace();
- uploadExceptionHandle(request,"保存上传文件时发生错误:"+e.getMessage());
- }
- if (forwardURL.length()==0){
- forwardURL=DEFAULT_UPLOAD_FAILURE_URL;
- }
- request.getRequestDispatcher(forwardURL).forward(request,response);
- }
{BackGroundService中的responseFileUploadStatusPoll方法用于处理对文件上传状态的轮询请求}:
java 代码
- /**
- * 回应上传状态查询
- * @param request
- * @param response
- * @throws IOException
- */
- private void responseFileUploadStatusPoll(HttpServletRequest request,HttpServletResponse response) throws IOException{
- response.setContentType("text/xml");
- response.setCharacterEncoding("UTF-8");
- response.setHeader("Cache-Control", "no-cache");
- logger.debug("发送上传状态回应");
- response.getWriter().write(XmlUnSerializer.serializeBean(
- request.getSession().getAttribute(UPLOAD_STATUS)));
- }
{BackGroundService中的processCancelFileUpload方法用于处理取消文件上传的请求}:
java 代码
- /**
- * 处理取消文件上传
- * @param request
- * @param response
- * @throws IOException
- */
- private void processCancelFileUpload(HttpServletRequest request,HttpServletResponse response) throws IOException{
- FileUploadStatus fUploadStatus=(FileUploadStatus)request.getSession().getAttribute(UPLOAD_STATUS);
- fUploadStatus.setCancel(true);
- request.getSession().setAttribute(UPLOAD_STATUS, fUploadStatus);
- responseFileUploadStatusPoll(request,response);
- }
Web客户端代码:
Prototype给开发者更多的自由选择
web客户端使用了基于Prototype的AjaxWrapper类和XMLDomForAjax类,前者实现了对Ajax.Request功能的封装,而后者实现了对来自服务器的XML Response的反序列化(反序列化为JavaScript对象)。
为了避免在AjaxWrapper的回调方法中发生this被重写的问题,我使用了ClassUtils类给任何类的每个方法注册一个对类对象自身引用,详见《解开JavaScript生命的达芬奇密码》和《Prototype.AjaxRequest的调用堆栈重写问题》:
{ClassUtils类代码}:
js 代码
- //类工具
- var ClassUtils=Class.create();
- ClassUtils.prototype={
- _ClassUtilsName:'ClassUtils',
- initialize:function(){
- },
- /**
- * 给类的每个方法注册一个对类对象的自我引用
- * @param reference 对类对象的引用
- */
- registerFuncSelfLink:function(reference){
- for (var n in reference) {
- var item = reference[n];
- if (item instanceof Function)
- item.$ = reference;
- }
- }
- }
{将XML反序列化为JavaScript对象的XMLDomForAjax类代码}:
js 代码
- var XMLDomForAjax=Class.create();
- XMLDomForAjax.prototype={
- isDebug:false,
- //dom节点类型常量
- ELEMENT_NODE:1,
- ATTRIBUTE_NODE:2,
- TEXT_NODE:3,
- CDATA_SECTION_NODE:4,
- ENTITY_REFERENCE_NODE:5,
- ENTITY_NODE:6,
- PROCESSING_INSTRUCTION_NODE:7,
- COMMENT_NODE:8,
- DOCUMENT_NODE:9,
- DOCUMENT_TYPE_NODE:10,
- DOCUMENT_FRAGMENT_NODE:11,
- NOTATION_NODE:12,
- initialize:function(isDebug){
- new ClassUtils().registerFuncSelfLink(this);
- this.isDebug=isDebug;
- },
- /**
- * 建立跨平台的dom解析器
- * @param xml xml字符串
- * @return dom解析器
- */
- createDomParser:function(xml){
- // code for IE
- if (window.ActiveXObject){
- var doc=new ActiveXObject("Microsoft.XMLDOM");
- doc.async="false";
- doc.loadXML(xml);
- }
- // code for Mozilla, Firefox, Opera, etc.
- else{
- var parser=new DOMParser();
- var doc=parser.parseFromString(xml,"text/xml");
- }
- return doc;
- },
- /**
- * 反向序列化xml到javascript Bean
- * @param xml xml字符串
- * @return javascript Bean
- */
- deserializedBeanFromXML:function (xml){
- var funcHolder=arguments.callee.$;
- var doc=funcHolder.createDomParser(xml);
- // documentElement总表示文档的root
- var objDomTree=doc.documentElement;
- var obj=new Object();
- for (var i=0; i
- //获得节点
- var node=objDomTree.childNodes[i];
- //取出其中的field元素进行处理
- if ((node.nodeType==funcHolder.ELEMENT_NODE) && (node.tagName == 'field')) {
- var nodeText=funcHolder.getNodeText(node);
- if (funcHolder.isDebug){
- alert(node.getAttribute('name')+' type:'+node.getAttribute('type')+' text:'+nodeText);
- }
- var objFieldValue=null;
- //如果为列表
- if (node.getAttribute('type')=='java.util.List'){
- if (objFieldValue && typeof(objFieldValue)=='Array'){
- if (nodeText.length>0){
- objFieldValue[objFieldValue.length]=nodeText;
- }
- }
- else{
- objFieldValue=new Array();
- }
- }
- else if (node.getAttribute('type')=='long'
- || node.getAttribute('type')=='java.lang.Long'
- || node.getAttribute('type')=='int'
- || node.getAttribute('type')=='java.lang.Integer'){
- objFieldValue=parseInt(nodeText);
- }
- else if (node.getAttribute('type')=='double'
- || node.getAttribute('type')=='float'
- || node.getAttribute('type')=='java.lang.Double'
- || node.getAttribute('type')=='java.lang.Float'){
- objFieldValue=parseFloat(nodeText);
- }
- else if (node.getAttribute('type')=='java.lang.String'){
- objFieldValue=nodeText;
- }
- else{
- objFieldValue=nodeText;
- }
- //赋值给对象
- obj[node.getAttribute('name')]=objFieldValue;
- if (funcHolder.isDebug){
- alert(eval('obj.'+node.getAttribute('name'))); </sp>
- AjaxFileUpload_u2.war (1.6 MB)
- 下载次数: 40
发表评论
-
主题:整理的DWR学习笔记(转)
2010-06-11 18:29 0DWR是什么,懒得说啦,上网百度一下吧, 学校差不多断网了,借 ... -
ajax jsp 无刷新上传文件
2009-10-18 18:08 2521是无法实现上传文件的,可以想一下ajax与后台通信都是通过传递 ... -
JSTL <fmt:formatDate/>
2009-02-26 09:38 2823关键字: fmt:formatdate, jstl fmt: ... -
Jakarta Commons HttpClient 学习笔记
2009-02-19 14:19 3276Jakarta Commons HttpClie ... -
post和get方法的区别总结
2009-02-19 11:56 3924post和get方法的区别总结 简 ... -
SiteMesh:一个优于Apache Tiles的Web页面布局、装饰框架
2009-02-17 10:06 2428SiteMesh:一个优于Apache ...
相关推荐
原文地址 http://www.telio.be/blog/2006/01/06/ajax-upload-progress-monitor-for-commons-fileupload-example/ 博文链接:https://congjl2002.iteye.com/blog/209925
本资源提供了一种实现大文件上传并带有进度条展示的解决方案,这对于用户交互体验有着显著的提升,尤其在处理大型文件时,能够实时反馈上传进度,增强用户对操作的掌控感。 【描述】:“带进度条的大文件上传源码...
嵌入式八股文面试题库资料知识宝典-华为的面试试题.zip
训练导控系统设计.pdf
嵌入式八股文面试题库资料知识宝典-网络编程.zip
人脸转正GAN模型的高效压缩.pdf
少儿编程scratch项目源代码文件案例素材-几何冲刺 转瞬即逝.zip
少儿编程scratch项目源代码文件案例素材-鸡蛋.zip
嵌入式系统_USB设备枚举与HID通信_CH559单片机USB主机键盘鼠标复合设备控制_基于CH559单片机的USB主机模式设备枚举与键盘鼠标数据收发系统支持复合设备识别与HID
嵌入式八股文面试题库资料知识宝典-linux常见面试题.zip
面向智慧工地的压力机在线数据的预警应用开发.pdf
基于Unity3D的鱼类运动行为可视化研究.pdf
少儿编程scratch项目源代码文件案例素材-霍格沃茨魔法学校.zip
少儿编程scratch项目源代码文件案例素材-金币冲刺.zip
内容概要:本文深入探讨了HarmonyOS编译构建子系统的作用及其技术细节。作为鸿蒙操作系统背后的关键技术之一,编译构建子系统通过GN和Ninja工具实现了高效的源代码到机器代码的转换,确保了系统的稳定性和性能优化。该系统不仅支持多系统版本构建、芯片厂商定制,还具备强大的调试与维护能力。其高效编译速度、灵活性和可扩展性使其在华为设备和其他智能终端中发挥了重要作用。文章还比较了HarmonyOS编译构建子系统与安卓和iOS编译系统的异同,并展望了其未来的发展趋势和技术演进方向。; 适合人群:对操作系统底层技术感兴趣的开发者、工程师和技术爱好者。; 使用场景及目标:①了解HarmonyOS编译构建子系统的基本概念和工作原理;②掌握其在不同设备上的应用和优化策略;③对比HarmonyOS与安卓、iOS编译系统的差异;④探索其未来发展方向和技术演进路径。; 其他说明:本文详细介绍了HarmonyOS编译构建子系统的架构设计、核心功能和实际应用案例,强调了其在万物互联时代的重要性和潜力。阅读时建议重点关注编译构建子系统的独特优势及其对鸿蒙生态系统的深远影响。
嵌入式八股文面试题库资料知识宝典-奇虎360 2015校园招聘C++研发工程师笔试题.zip
嵌入式八股文面试题库资料知识宝典-腾讯2014校园招聘C语言笔试题(附答案).zip
双种群变异策略改进RWCE算法优化换热网络.pdf
内容概要:本文详细介绍了基于瞬时无功功率理论的三电平有源电力滤波器(APF)仿真研究。主要内容涵盖并联型APF的工作原理、三相三电平NPC结构、谐波检测方法(ipiq)、双闭环控制策略(电压外环+电流内环PI控制)以及SVPWM矢量调制技术。仿真结果显示,在APF投入前后,电网电流THD从21.9%降至3.77%,显著提高了电能质量。 适用人群:从事电力系统研究、电力电子技术开发的专业人士,尤其是对有源电力滤波器及其仿真感兴趣的工程师和技术人员。 使用场景及目标:适用于需要解决电力系统中谐波污染和无功补偿问题的研究项目。目标是通过仿真验证APF的有效性和可行性,优化电力系统的电能质量。 其他说明:文中提到的仿真模型涉及多个关键模块,如三相交流电压模块、非线性负载、信号采集模块、LC滤波器模块等,这些模块的设计和协同工作对于实现良好的谐波抑制和无功补偿至关重要。