文件上传下载
使用spring-mvc,完成文件上传与下载功能
jar包依赖
<!--文件上传--> <dependency> <groupId>commons-io</groupId> <artifactId>commons-io</artifactId> <version>2.6</version> </dependency> <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>1.3.2</version> </dependency>
spring-mvc.xml文件配置
<!--文件上传--> <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> <property name="defaultEncoding" value="UTF-8"/> <property name="maxUploadSize" value="#{10*1024*1024}"/> </bean>
上传
文件上传分为三种:单文件上传、多文件上传、异步上传
单文件上传
HTML代码
<form method="post" action="../testUpload" enctype="multipart/form-data"> <input type="file" name="file"/> <input type="submit" value="submit"/> </form>
注意:
- form表单中要设置enctype属性为multipart/form-data,使用二进制流的形式发送数据
- 用于上传文件的input要设置type为file
后台中使用MultipartFile参数即可接收传递过来的二进制文件数据
Java代码
@RequestMapping("/upload") public String upload(@RequestParam("file1") MultipartFile file, Model model, HttpServletRequest request) throws IOException { if (!file.isEmpty()) { // 获取文件路径 String path = request.getSession().getServletContext().getRealPath("/static/upload"); // 获取原始文件名 String filename = file.getOriginalFilename(); // 使用时间戳重新命名文件,避免文件重名 SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyyMMddHHmmss"); String prefix = simpleDateFormat.format(new Date()); String suffix = filename.substring(filename.lastIndexOf(".")); String newFileName = prefix + suffix; // 创建目标文件对象 File descFile = new File(path + "/" + newFileName); // 如果目标路径不存在 if (!descFile.getParentFile().exists()) { descFile.mkdirs(); } // 如果目标文件不存在 if (!descFile.exists()) { file.transferTo(descFile); model.addAttribute("file", newFileName); } } return "front/ok";}
多文件上传
多文件上传的思路就是将多个文件分开,对每个文件使用单文件上传的方法即可。但是这其中又分为两种情况:一为在同一个文件框中一次选择多个文件上传,二为在多个文件框中同时选择文件上传
情况一 单个文件框内选择多个文件
此情况与单文件上传相比需要在input中添加multiple属性,它可以使我们在这个input框中同时选择多个文件上传
HTML代码
<form method="post" action="../testUpload2" enctype="multipart/form-data"> <input type="file" name="file" multiple="multiple"/> <input type="submit" value="submit"/> </form>
与单文件上传相比,后台中我们只需要将MultipartFile参数改为数组形式即可接收传递来的一组二进制文件数据,之后对这个数组进行遍历使用单文件上传的方法即可。
Java代码
@RequestMapping("/testUpload2") public String upload2(HttpSession session, @RequestParam("file") MultipartFile[] multipartFiles) throws IOException { int i = 1; if (multipartFiles != null && multipartFiles.length > 0) { String path = session.getServletContext().getRealPath("/static/upload"); for (MultipartFile file : multipartFiles) { String filename = file.getOriginalFilename(); SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyyMMddHHmmss"); String prefix = simpleDateFormat.format(new Date()); String suffix = filename.substring(filename.lastIndexOf(".")); String newFileName = prefix + i + suffix; File descFile = new File(path + "/" + newFileName); if (!descFile.getParentFile().exists()) { descFile.mkdirs(); } if (!descFile.exists()) { file.transferTo(descFile); } i++; } } return "front/ok"; }
情况二 多个文件框同时上传文件
form表单中同时存在多个有相同name值的file类型input
HTML代码
<form method="post" action="../testUpload" enctype="multipart/form-data"> <input type="file" name="file"/> <input type="file" name="file"/> <input type="submit" value="submit"/> </form>
此种情况下我们在后台不能再使用MultipartFile来接受文件,而要使用MultipartRequest参数来进行文件获取。
Java代码
@RequestMapping("/testUpload") public String upload(HttpSession session, MultipartRequest multipartRequest) throws IOException { // 使用MultipartRequest.getFiles()方法获取所有同名文件框中的文件 List<MultipartFile> files = multipartRequest.getFiles("file"); int i = 1; if ( files.size() > 0 ) { String path = session.getServletContext().getRealPath("/static/upload"); for (MultipartFile file : files) { String filename = file.getOriginalFilename(); SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyyMMddHHmmss"); String prefix = simpleDateFormat.format( new Date() ); String suffix = filename.substring(filename.lastIndexOf(".")); String newFileName = prefix + i + suffix; File descFile = new File( path + "/" + newFileName ); if ( !descFile.getParentFile().exists() ) { descFile.mkdirs(); } if ( !descFile.exists() ) { file.transferTo(descFile); } i++; } } return "front/ok"; }
异步上传
异步上传与前两种方式相比,区别在于使用ajax技术向后台发送文件
JavaScript代码
function uploadData(){ var fdata = new FormData(mygoodsform);//提取表单数据 //通过 ajax将数据发送到 后台 $.ajax({ url:"/EasyUIDemo/emp/uploadData.do", type:"post", data:fdata, processData:false, contentType:false, success:function(resp){ // do something } }); }
提取表单数据的两种方式:
- js方式 new FormData(表单id)
- jQuery方式 new FormData($("#表单id")[0])
两个关键属性:
processData:
要求为Boolean类型的参数,默认为true。默认情况下,发送的数据将被转换为对象(从技术角度来讲并非字符串)以配合默认内容类 型"application/x-www-form-urlencoded"。如果要发送DOM树信息或者其他不希望转换的信息,请设置为false。
contentType:
要求为String类型的参数,当发送信息至服务器时,内容编码类型默认为"application/x-www-form-urlencoded"。该默认值适合大多数应用场合。
下载
下载有两种方式:其一为通过a标签指向文件路径进行下载,其二为通过后台代码控制浏览器进行下载。
方式一 a标签
HTML代码
<a href="../static/upload/111.sql">下载</a>
此种方式有诸多缺点:
- 如果目标文件是图片类型,则会直接打开图片,需要右击超链接点击另存为进行下载
- 容易出现中文乱码问题
不能进行多文件下载
方式二 后台代码控制浏览器下载(多文件下载)
单文件下载
Java代码
@RequestMapping("/down") public ResponseEntity<byte[]> download(@RequestParam("filename") String filename, HttpServletRequest request) throws IOException { String path = request.getSession().getServletContext().getRealPath("/static/upload"); File file = new File(path + "/" + filename); //文件名(中文)转码 String encodedFileName = null; String userAgentString = request.getHeader("User-Agent"); if (userAgentString.indexOf("Chrome") > 0 || userAgentString.indexOf("FireFox") > 0) { encodedFileName = URLEncoder.encode(filename, "UTF-8").replaceAll("\\+","%20"); } else { encodedFileName = MimeUtility.encodeWord(filename); } //数据,将要下载的文件对象转换成字节数组 byte[] data = FileUtils.readFileToByteArray(file); HttpHeaders headers = new HttpHeaders(); //设置响应文档类型,二进制形式 headers.setContentType(MediaType.APPLICATION_OCTET_STREAM); //弹出下载窗口,并指定文件名 headers.setContentDispositionFormData("attachment", encodedFileName); //状态码 HttpStatus status = HttpStatus.CREATED; ResponseEntity<byte[]> responseEntity = new ResponseEntity<byte[]>(data, headers, status); return responseEntity; }
多文件下载:将多个文件先打包成zip文件,再使用单文件下载的方法下载这个zip即可
Java代码
@RequestMapping("downloadHomeworks.do") public void downloadHomewords(String paths, HttpServletRequest request, HttpServletResponse response) { String tmpFileName = "homeworks.zip"; byte[] buffer = new byte[1024]; String strZipPath = "D:/hw/" + UUID.randomUUID() + tmpFileName; try { ZipOutputStream out = new ZipOutputStream(new FileOutputStream(strZipPath)); String replace = paths.replace(',', ' '); String[] files = replace.split(" "); // 下载的文件集合 for (int i = 0; i < files.length; i++) { FileInputStream fis = new FileInputStream(files[i]); out.putNextEntry(new ZipEntry(files[i].substring(6))); // 设置压缩文件内的字符编码,不然会变成乱码 // out.setEncoding("GBK"); int len; // 读入需要下载的文件的内容,打包到zip文件 while ((len = fis.read(buffer)) > 0) { out.write(buffer, 0, len); } out.closeEntry(); fis.close(); } out.flush(); out.close(); // 调用单文件下载的方法下载打包好的压缩文件 downloadHomeword(strZipPath, request, response); } catch (Exception e) { e.printStackTrace(); } }