springMVC的上传和下载 Day33 2018-12-23

springMVC的上传和下载

1. 文件上传

上传文件必须将表单的method设置为post,并且将enctype设置为multipart/form-data

springmvc中实现文件上传有两种方式:

1、使用Apache Commons FileUpload元件。

2、利用servlet3.0及其更高版本的内置支持。

  • MultipartFile常用方法:
返回参数 方法
byte[] getBytes() 返回文件的内容作为一个字节数组。
String getContentType() 返回文件的内容类型。
InputStream getInputStream() 返回InputStream读取文件的内容。
String getName() 返回表单中文件组件的名字。
String getOriginalFilename() 返回上传文件的原名
long getSize() 返回文件的大小,以字节byte为单位。
boolean isEmpty() 返回是否上传文件是空的,也就是说,没有文件 选择多部分形式或所选的文件中没有的内容。
void transferTo(File dest) 接收到的文件转移到给定的目标文件。

1.1 配置tomcat虚拟路径

编辑server文件(%tomcathome%\conf\server.xml)
我们打算建立一个images的虚拟目录,只要在%tomcathome%\conf\server.xml文件,在<host>标签中加入文件中加入如下代码即可:

<Context debug="0" docBase="D:\APPFile\images" path="/images" reloadable="true"/>

注意,此时path一定要写,因为我们此时没有新建xml文件了,所以一定要指明web。

配置文件属性的含义:

  • docBase为希望将文件存储到的物理目录的绝对路径。

  • path为访问文件时的虚拟目录。

  • reloadable属性的设置有些用处,当reloadable=true时,相关文件改变,Tomcat先停止webapp并释放内存,然后重新加载webapp。这样以来可以省去手工部署web app工程的时间。和开发工具一起使用可以稍微提高点工作效。

2.使用 Apache Commons FileUpload元件上传

2.1 添加jar包

<dependency>
        <groupId>commons-fileupload</groupId>
        <artifactId>commons-fileupload</artifactId>
        <version>1.3.3</version>
    </dependency>
    <dependency>
        <groupId>commons-io</groupId>
        <artifactId>commons-io</artifactId>
        <version>2.6</version>
    </dependency>

2.2 springmvc配置

spring MVC 上下文中没有装配multipartResolver,不能处理文件上传

bean id="multipartResolver"  
        class="org.springframework.web.multipart.commons.CommonsMultipartResolver">  
        <!-- 上传文件大小上限,单位为字节(10MB) -->
        <property name="maxUploadSize">  
            <value>10485760</value>  
        </property>  
        <!-- 请求的编码格式,必须和jSP的pageEncoding属性一致,以便正确读取表单的内容,默认为ISO-8859-1 -->
        <property name="defaultEncoding">
            <value>UTF-8</value>
        </property>
    </bean>

2.3 Controller

@Controller
@RequestMapping(value = "/file")
public class FileUploadController {
    @RequestMapping(value = "/{formName}")
    public String loginForm(@PathVariable String formName) {
        // 动态跳转页面
        return "file/" + formName;
    }

    // 上传文件会自动绑定到MultipartFile中
    @RequestMapping(value = "/upload", method = RequestMethod.POST)
    public String upload(HttpServletRequest request, @RequestParam("description") String description,
            @RequestParam("file") MultipartFile file) throws Exception {

        System.out.println(description);
        // 如果文件不为空,写入上传路径
        if (!file.isEmpty()) {
            // 上传文件路径
            // String path = request.getServletContext().getRealPath("/images/");
            //图片服务器路径
            String path = "D:\\QYAPP\\images\\";
            // 上传文件名 原始文件名
            String filename = file.getOriginalFilename();
            //创建新文件,路径为:图片服务器路径+新文件名
            File filepath = new File(path, filename);
            // 判断路径是否存在,如果不存在就创建一个
            if (!filepath.getParentFile().exists()) {
                filepath.getParentFile().mkdirs();
            }
            // 将上传文件保存到一个目标文件当中
            file.transferTo(new File(path + File.separator + filename));
            return "file/" + "success";
        } else {
            return "file/" + "error";
        }

    }
}

2.4 jsp

  • uploadForm.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>文件上传</title>
</head>
<body>
    <h2>文件上传</h2>
    <form action="upload" enctype="multipart/form-data" method="post">
        <table>
            <tr>
                <td>文件描述:</td>
                <td><input type="text" name="description"></td>
            </tr>
            <tr>
                <td>请选择文件:</td>
                <td><input type="file" name="file"></td>
            </tr>
            <tr>
                <td><input type="submit" value="上传"></td>
            </tr>
        </table>
    </form>
</body>
</html>

3. 使用对象接收上传文件

3.1 实体类

public class User  implements Serializable{

    private String username;
    private MultipartFile image;
    private String path;
 //省略get/set方法   
}

3.2 Controller层

@RequestMapping(value="/register")
     public String register(HttpServletRequest request,
             @ModelAttribute User user,
             Model model)throws Exception{
        System.out.println(user.getUsername());
        // 如果文件不为空,写入上传路径
        if(!user.getImage().isEmpty()){
            // 上传文件路径
            //String path = request.getServletContext().getRealPath("/images/");
            //图片服务器路径
            String path = "D:\\QYAPP\\images\\";
            // 上传文件名
            String filename = user.getImage().getOriginalFilename();
            File filepath = new File(path,filename);
            // 判断路径是否存在,如果不存在就创建一个
            if (!filepath.getParentFile().exists()) { 
                filepath.getParentFile().mkdirs();
            }
            // 将上传文件保存到一个目标文件当中
            user.getImage().transferTo(new File(path+File.separator+ filename));
            //虚拟图片路径
            String imagePath = request.getRequestURL().toString();
            imagePath = imagePath.substring(0, getIndexAtStr(imagePath, "/", 3))+"images"+File.separator+ filename; 
            System.out.println(imagePath);
            user.setPath(imagePath);
            // 将用户添加到model
            model.addAttribute("user", user);
            return "file/" + "userInfo";
        }else{
            return "file/" + "error";
        }
    }
    /**
     * 返回字符出现第num次的位置索引
     * @param str 字符串
     * @param key 指定字符串
     * @param num 在字符串中出现的次数
     * @return
     */
     public static int getIndexAtStr(String str,String key,int num){
            int count = 0;
            int index = 0;
            while((index=str.indexOf(key,index))!=-1){
                //System.out.println("index="+index);
                index = index+key.length();//根据在字符串中出现的位置,计数一次,下次从该位置后重新查找出现新的位置
                count++;
                if(count==num) break;//第几次跳槽循环
            }
            //return count;
            return index;
        }

3.3 jsp

3.3.1 registerForm.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>用户注册</title>
</head>
<body>
    <h2>用户注册</h2>
    <form action="register" enctype="multipart/form-data" method="post">
        <table>
            <tr>
                <td>用户名:</td>
                <td><input type="text" name="username"></td>
            </tr>
            <tr>
                <td>请上传头像:</td>
                <td><input type="file" name="image"></td>
            </tr>
            <tr>
                <td><input type="submit" value="注册"></td>
            </tr>
        </table>
    </form>
</body>
</html>
3.3.2 userInfo.jsp
<a href="download?filename=${requestScope.user.image.originalFilename}">
${requestScope.user.image.originalFilename }
</a>
<img alt="${requestScope.user.image.originalFilename }" src="${requestScope.user.path }">

4. Spring MVC 文件下载

页面给出 超链接,href的属性等于下载文件的路径,即可下载

springMVC 提供了一个ResponseEntity类型,可以定义返回的HttpHeadersHttpStatus

4.1 Controller层

@RequestMapping(value="/download")
     public ResponseEntity<byte[]> download(HttpServletRequest request,
             @RequestParam("filename") String filename,
             Model model)throws Exception{
        // 下载文件路径
        //String path = request.getServletContext().getRealPath("/images/");
        //图片服务器路径
        String path = "D:\\QYAPP\\images\\";
        File file = new File(path+File.separator+ filename);
        HttpHeaders headers = new HttpHeaders();  
        // 下载显示的文件名,解决中文名称乱码问题  
        String downloadFielName = new String(filename.getBytes("UTF-8"),"iso-8859-1");
        // 通知浏览器以attachment(下载方式)打开图片
        headers.setContentDispositionFormData("attachment", downloadFielName); 
        // application/octet-stream : 二进制流数据(最常见的文件下载)。
        headers.setContentType(MediaType.APPLICATION_OCTET_STREAM);
        // 201 HttpStatus.CREATED
        return new ResponseEntity<byte[]>(FileUtils.readFileToByteArray(file),    
                headers, HttpStatus.CREATED);  
     }

5. 利用servlet3.0及其更高版本的内置支持上传文件

有了Servlet3,就不再需要Commons FileUploadCommons IO元件了。在servlet3及以上版本的容器进行上传文件,处理已上传文件的Servlets必须以@MultipartConfig进行标注。下列是可能在MultipartConfig标注类型中出现的属性,它们都是可选的:

  • maxFileSize:上传问件的最大容量,默认值为-1,表示没有限制。大于指定值的文件将遭到拒绝。
  • maxRequestSize:表示多部分HTTP请求允许的最大容量,默认值为-1。
  • location:表示在part调用write方法时,要将已上传文件保存的磁盘中的位置。
  • fileSizeThreshold:上传文件超出这个容量界限时,会被写入磁盘。

5.1 配置web.xml

(1)web.xml
Spring MVCDispatcherServlet处理大部分或所有请求,如果要以@MultipartConfig进行标注,需要修改源代码。这是不可取的,不过,Servlet3中有一个比较容易的方法,能使一个servlet变成一个MultipartConfig Servlet,即给部署描述符(web.xml)中的servlet声明赋值。以下代码与用@MultipartConfigDispatcherServlet进行标注的效果一样。

<!-- springMVC配置 前端控制器 -->
    <servlet>
        <servlet-name>springmvc</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:springmvc-config.xml</param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
        <!-- 用servlet3及其更高版本上传文件,给DispatcherServlet添加MultipartConfig标注 -->
        <multipart-config>
            <max-file-size>20848820</max-file-size>
            <max-request-size>418018841</max-request-size>
        </multipart-config>
    </servlet>

5.2 springmvc配置

 <!-- 用servlet3及其更高版本上传文件,使用一个不同的多部分解析器multipartResolver -->
     <bean id="multipartResolver" class="org.springframework.web.multipart.support.StandardServletMultipartResolver"/>

5.3 Controller层(多图上传)

@RequestMapping(value="/multiUpload")
     public String multiUpload(HttpServletRequest request, @ModelAttribute User user,//@RequestParam("username") String username,
             @RequestParam("files") MultipartFile[] files,
             Model model)throws Exception{
        /*User user = new User();
        user.setUsername(username);*/
        List<String> listImagePath=new ArrayList<String>();

        /*List<MultipartFile> files = null;
        //得到多部分解析器
        CommonsMultipartResolver multipartResolver = new CommonsMultipartResolver(request.getSession().getServletContext());
        //判断request是否有文件上传,即多部分
        if(multipartResolver.isMultipart(request)){
            //转换为多部分request
            MultipartHttpServletRequest multipartRequest = (MultipartHttpServletRequest) request;
            //取得multipartRequest中的所有文件
            files = multipartRequest.getFiles("files");
        }*/
        //虚拟图片路径
        String imagePath = request.getRequestURL().toString();
        imagePath = imagePath.substring(0, getIndexAtStr(imagePath, "/", 3))+"images/"; 
        /** 图片按类别存储 */
        /** 支持多图片上传 */
        //if(files != null && files.size() > 0){
        if(files != null && files.length>0) {
            int i = 0;
            //图片服务器路径
            String file_path = "D:\\QYAPP\\images\\";
            String filename = "jpg";
            //根据类别名创建文件夹
            File dir = new File(file_path + filename);
            if(!dir.exists() || !dir.isDirectory()){
                dir.mkdir();
            }
            for(MultipartFile file : files){
                if(file != null && file.getOriginalFilename() != null && file.getOriginalFilename().length()>0){
                    //原始文件名
                    String originalFileName = file.getOriginalFilename();
                    //新文件名,添加原始文件名后缀
                    String newFileName = UUID.randomUUID() + originalFileName.substring(originalFileName.lastIndexOf("."));
                    //创建新文件,路径为:图片服务器路径+文件夹名+新文件名
                    File newFile = new File(file_path + filename + "\\" + newFileName);
                    //将内存中的数据写入磁盘
                    file.transferTo(newFile);
                    i++;
                    listImagePath.add(imagePath+ filename + "\\" + newFileName);
                    user.setImage(file);
                }
            }
            // 将用户添加到model
            model.addAttribute("listImagePath", listImagePath);
            user.setPath(listImagePath.get(0));
            model.addAttribute("user", user);
            return "file/" + "userInfo";
        }else{
            return "file/" + "error";
        }
    }

5.4 jsp

5.4.1 multiUploadForm.jsp
<form action="multiUpload" enctype="multipart/form-data" method="post">
        <table>
            <tr>
                <td>文件描述:</td>
                <td><input type="text" name="username"></td>
            </tr>
            <tr>
                <td>请选择文件:</td>
                <td><input type="file" name="files"></td>
                <td><input type="file" name="files"></td>
                <td><input type="file" name="files"></td>
            </tr>
            <tr>
                <td><input type="submit" value="上传"></td>
            </tr>
        </table>
    </form>
5.4.2 userInfo.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>文件下载</title>
</head>
<body>
<h3>文件下载</h3>
<a href="download?filename=${requestScope.user.image.originalFilename}">
${requestScope.user.image.originalFilename }
</a>
<img alt="${requestScope.user.image.originalFilename }" src="${requestScope.user.path }">


<c:forEach items="${listImagePath}" var="image">
    <img src="${image}"><br/>
</c:forEach>

</body>
</html>

6.总结

以上两种方式,在pojo类和controller类以及jsp中的代码并无不同,只是第二种方式需要在web.xml中对DispatcherServlet添加multipart-config标注,并且两种方法在springmvc中配置的多部分解析器的bean不同。其余代码无需改变。

  • 多图上传见 5.3
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容