java webApp异步上传图片实现代码
如何实现javawebApp异步上传图片,先了解以下几个问题:
1.图片上传;
2.图片上传预览;
3.上传图片更改地址异步添加到数据库;
主要内容
本示例主要采用纯HTML前端和JavaScript代码作工具,查询有关demo其实现图片上传的代码范例如下:
(1)点击上传图片的div代码:
<divid="div1"class="photo"> <inputtype="file"id="choose"accept="image/*"multiple> <aid="upload">上传图片</a> <aclass="myinputimg"onclick="selectphoto();">从图库中选择</a> <aclass="myinputimg"id="back">取消</a> </div>
(2)javaScript代码
<scripttype="text/javascript"> //获取上传图片的input表单元素 varfilechooser=document.getElementById("choose"); //创建用于压缩图片的canvas varcanvas=document.createElement("canvas"); //获取canvas的视觉属性 varctx=canvas.getContext('2d'); //瓦片canva vartCanvas=document.createElement("canvas"); vartctx=tCanvas.getContext("2d"); //画布的大小 varmaxsize=100*1024; //上传图片点击事件 $("#upload").on("click",function(){ filechooser.click(); }) .on("touchstart",function(){ //添加元素属性 $(this).addClass("touch"); }) .on("touchend",function(){ //移除元素属性 $(this).removeClass("touch"); }); //元素改变 filechooser.onchange=function(){ //如果选择为空,返回操作 if(!this.files.length)return; //创建上传图片的数组 varfiles=Array.prototype.slice.call(this.files); //选择为数量大于1张时,反回操作,这里根据需求设定;pc端测试一次可以上传若干张图片,移动端选择一张,页面只能预览一张。由于是移动端,所以作此判断。 if(files.length>1){ alert("一次只能上传1张图片"); return; } //遍历上传图片的文件数组,可不用遍历,直接取即可。 files.forEach(function(file,i){ //判断图片格式 if(!/\/(?:jpeg|png|gif)/i.test(file.type))return; varreader=newFileReader(); varli=document.createElement("li"); //获取图片大小 varsize=file.size/1024>1024?(~~(10*file.size/1024/1024))/10+"MB":~~(file.size/1024)+"KB"; //图片预览 li.innerHTML='<divclass="progress"><span></span></div><divclass="size">'+size+'</div>'; //追加图片预览代码; $(".img-list").append($(li)); reader.onload=function(){ varresult=this.result; varimg=newImage(); img.src=result; //图片显示 $(li).css("background-image","url("+result+")"); //如果图片大小小于100kb,则直接上传 if(result.length<=maxsize){ img=null; upload(result,file.type,$(li)); return; } //图片加载完毕之后进行压缩,然后上传 if(img.complete){ callback(); }else{ img.onload=callback; } functioncallback(){ vardata=compress(img); upload(data,file.type,$(li)); img=null; } }; reader.readAsDataURL(file); }); }; //以下是图片压缩相关; //使用canvas对大图片进行压缩 functioncompress(img){ varinitSize=img.src.length; varwidth=img.width; varheight=img.height; //如果图片大于四百万像素,计算压缩比并将大小压至400万以下 varratio; if((ratio=width*height/4000000)>1){ ratio=Math.sqrt(ratio); width/=ratio; height/=ratio; }else{ ratio=1; } canvas.width=width; canvas.height=height; //铺底色 ctx.fillStyle="#fff"; ctx.fillRect(0,0,canvas.width,canvas.height); //如果图片像素大于100万则使用瓦片绘制 varcount; if((count=width*height/1000000)>1){ count=~~(Math.sqrt(count)+1);//计算要分成多少块瓦片 //计算每块瓦片的宽和高 varnw=~~(width/count); varnh=~~(height/count); tCanvas.width=nw; tCanvas.height=nh; for(vari=0;i<count;i++){ for(varj=0;j<count;j++){ tctx.drawImage(img,i*nw*ratio,j*nh*ratio,nw*ratio,nh*ratio,0,0,nw,nh); ctx.drawImage(tCanvas,i*nw,j*nh,nw,nh); } } }else{ ctx.drawImage(img,0,0,width,height); } //进行最小压缩 varndata=canvas.toDataURL('image/jpeg',0.1); console.log('压缩前:'+initSize); console.log('压缩后:'+ndata.length); console.log('压缩率:'+~~(100*(initSize-ndata.length)/initSize)+"%"); tCanvas.width=tCanvas.height=canvas.width=canvas.height=0; returnndata; } //图片上传,将base64的图片转成二进制对象,塞进formdata上传 functionupload(basestr,type,$li){ vartext=window.atob(basestr.split(",")[1]); varbuffer=newUint8Array(text.length); varpecent=0,loop=null; for(vari=0;i<text.length;i++){ buffer[i]=text.charCodeAt(i); } varblob=getBlob([buffer],type); varxhr=newXMLHttpRequest(); varformdata=getFormData(); formdata.append('upload',blob); //异步请求kindeditor插件的上传图片jsp页面 xhr.open('post','<%=request.getContextPath()%>/kindeditor/jsp/upload_json.jsp'); xhr.onreadystatechange=function(){ if(xhr.readyState==4&&xhr.status==200){ //返回服务器端的图片地址 varface_img=xhr.responseText; varid=$("#arId").text(); //异步像数据库中添加图片 $.ajax({ type:"POST", //异步请求Struts的action类将图片地址插入数据库 url:"add_article_faceurl.action", dataType:"json", data:"faceurl="+face_img+"&id="+id, async:true, success:function(msg){ //取添加数据库中的图片相关的id值,存入页面隐藏区域 $("#arId").text(msg); }, error:function(a){} }); } }; //模拟上传进度显示 //数据发送进度,前50%展示该进度 xhr.upload.addEventListener('progress',function(e){ if(loop)return; pecent=~~(100*e.loaded/e.total)/2; $li.find(".progressspan").css('width',pecent+"%"); if(pecent==50){ mockProgress(); } },false); //数据后50%用模拟进度 functionmockProgress(){ if(loop)return; loop=setInterval(function(){ pecent++; $li.find(".progressspan").css('width',pecent+"%"); if(pecent==99){ clearInterval(loop); } },100); } xhr.send(formdata); } /** *获取blob对象的兼容性写法 *@parambuffer *@paramformat *@returns{*} */ functiongetBlob(buffer,format){ try{ returnnewBlob(buffer,{type:format}); }catch(e){ varbb=new(window.BlobBuilder||window.WebKitBlobBuilder||window.MSBlobBuilder); buffer.forEach(function(buf){ bb.append(buf); }); returnbb.getBlob(format); } } /** *获取formdata */ functiongetFormData(){ varisNeedShim=~navigator.userAgent.indexOf('Android') &&~navigator.vendor.indexOf('Google') &&!~navigator.userAgent.indexOf('Chrome') &&navigator.userAgent.match(/AppleWebKit\/(\d+)/).pop()<=534; returnisNeedShim?newFormDataShim():newFormData(); } /** *formdata补丁,给不支持formdata上传blob的android机打补丁 *@constructor */ functionFormDataShim(){ console.warn('usingformdatashim'); varo=this, parts=[], boundary=Array(21).join('-')+(+newDate()*(1e16*Math.random())).toString(36), oldSend=XMLHttpRequest.prototype.send; this.append=function(name,value,filename){ parts.push('--'+boundary+'\r\nContent-Disposition:form-data;name="'+name+'"'); if(valueinstanceofBlob){ parts.push(';filename="'+(filename||'blob')+'"\r\nContent-Type:'+value.type+'\r\n\r\n'); parts.push(value); } else{ parts.push('\r\n\r\n'+value); } parts.push('\r\n'); }; //OverrideXHRsend() XMLHttpRequest.prototype.send=function(val){ varfr, data, oXHR=this; if(val===o){ //Appendthefinalboundarystring parts.push('--'+boundary+'--\r\n'); //Createtheblob data=getBlob(parts); //Setupandreadtheblobintoanarraytobesent fr=newFileReader(); fr.onload=function(){ oldSend.call(oXHR,fr.result); }; fr.onerror=function(err){ throwerr; }; fr.readAsArrayBuffer(data); //Setthemultipartcontenttypeandboudary this.setRequestHeader('Content-Type','multipart/form-data;boundary='+boundary); XMLHttpRequest.prototype.send=oldSend; } else{ oldSend.call(this,val); } }; } </script>
(3)kindeditor插件的上传图片jsp页面相关代码.
<%@pagelanguage="java"contentType="text/html;charset=UTF-8"pageEncoding="UTF-8"%> <%@pageimport="java.util.*,java.io.*"%> <%@pageimport="java.text.SimpleDateFormat"%> <%@pageimport="org.apache.commons.fileupload.*"%> <%@pageimport="org.apache.commons.fileupload.disk.*"%> <%@pageimport="org.apache.commons.fileupload.servlet.*"%> <%@page import="org.apache.struts2.dispatcher.multipart.MultiPartRequestWrapper"%> <%@pageimport="org.json.simple.*"%> <% /** *KindEditorJSP * *本JSP程序是演示程序,建议不要直接在实际项目中使用。 *如果您确定直接使用本程序,使用之前请仔细确认相关安全设置。 * */ //文件保存目录路径 StringsavePath=pageContext.getServletContext().getRealPath("/")+"attached/"; //StringsavePath="http:\\\\192.168.1.226:8080\\qslnbase\\uploadFile/"; //StringsavePath="D:/WWW/qslnADP/ADP/WebRoot/kindeditor/attached/"; //文件保存目录URL StringsaveUrl=request.getContextPath()+"/attached/"; //定义允许上传的文件扩展名 HashMap<String,String>extMap=newHashMap<String,String>(); extMap.put("image","gif,jpg,jpeg,png,bmp,blob"); extMap.put("flash","swf,flv"); extMap.put("media","swf,flv,mp3,wav,wma,wmv,mid,avi,mpg,asf,rm,rmvb"); extMap.put("file","doc,docx,xls,xlsx,ppt,htm,html,txt,zip,rar,gz,bz2"); //最大文件大小 longmaxSize=1000000; response.setContentType("text/html;charset=UTF-8"); if(!ServletFileUpload.isMultipartContent(request)){ out.println(getError("请选择文件。")); return; } //检查目录 FileuploadDir=newFile(savePath); if(!uploadDir.isDirectory()){ out.println(getError("上传目录不存在。")); return; } //检查目录写权限 if(!uploadDir.canWrite()){ out.println(getError("上传目录没有写权限。")); return; } StringdirName=request.getParameter("dir"); if(dirName==null){ dirName="image"; } if(!extMap.containsKey(dirName)){ out.println(getError("目录名不正确。")); return; } //创建文件夹 savePath+=dirName+"/"; saveUrl+=dirName+"/"; FilesaveDirFile=newFile(savePath); if(!saveDirFile.exists()){ saveDirFile.mkdirs(); } SimpleDateFormatsdf=newSimpleDateFormat("yyyyMMdd"); Stringymd=sdf.format(newDate()); savePath+=ymd+"/"; saveUrl+=ymd+"/"; FiledirFile=newFile(savePath); if(!dirFile.exists()){ dirFile.mkdirs(); } //Struts2请求包装过滤器 MultiPartRequestWrapperwrapper=(MultiPartRequestWrapper)request; //获得上传的文件名 StringfileName1=wrapper.getFileNames("upload")[0]; //获得文件过滤器 Filefile=wrapper.getFiles("upload")[0]; //检查文件大小 if(file.length()>maxSize){ out.println(getError("上传文件大小超过限制。")); return; } //检查扩展名 StringfileExt1=fileName1.substring(fileName1.lastIndexOf(".")+1).toLowerCase(); //重构上传文件名 SimpleDateFormatdf1=newSimpleDateFormat("yyyyMMddHHmmss"); StringnewFileName1=df1.format(newDate())+"_"+newRandom().nextInt(1000)+"."+fileExt1; byte[]buffer=newbyte[1024]; //获取文件输出流 FileOutputStreamfos=newFileOutputStream(savePath+newFileName1); Stringurl=savePath+newFileName1; out.println(url); //获取内存中当前文件输入流 InputStreamin=newFileInputStream(file); try{ intnum=0; while((num=in.read(buffer))>0){ fos.write(buffer,0,num); } }catch(Exceptione){ e.printStackTrace(System.err); }finally{ in.close(); fos.close(); } %> <%! privateStringgetError(Stringmessage){ JSONObjectobj=newJSONObject(); obj.put("error",1); obj.put("message",message); returnobj.toJSONString(); } %>
(4)有关kindeditor上传图片的jar包有如下所示
A.commons-fileupload-1.2.1.jar
B.commons-io-1.4.jar
C.json_simple-1.1.jar
这里没有用到有关于kindeditor的js代码,具体可参考:Kindeditor实现图片自动上传功能
(5)有关kindeditor上传图片预览的div如下
<divid="div2"> <ulclass="img-list"> <liid="wy"> <imgstyle="height:100%;width:100%;position:absolute;top:0px;"src="<%=request.getContextPath()%>/shequ/images/index.png;"> </li> </ul> </div>
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。