spring-mvc

乱码问题

SpringMVC 提供的过滤器

解决乱码问题:SpringMVC 给我们提供了一个过滤器,可以在 web.xml 中配置:

<filter>
   <filter-name>encoding</filter-name>
   <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
   <init-param>
       <param-name>encoding</param-name>
       <param-value>utf-8</param-value>
   </init-param>
</filter>
<filter-mapping>
   <filter-name>encoding</filter-name>
   <url-pattern>/*</url-pattern>
</filter-mapping>

但是有些极端情况下,这个过滤器对 get 的支持不好。

处理方法:修改 tomcat 配置文件,设置编码:

<Connector URIEncoding="utf-8" port="8080" protocol="HTTP/1.1"
          connectionTimeout="20000"
          redirectPort="8443" />

或者可以在代码中放入如下的 自定义过滤器

自定义过滤器

package com.yusael.filter;

import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.util.Map;

/**
 * 解决 get 和 post 请求 全部乱码的过滤器
 */
public class GenericEncodingFilter implements Filter {

    @Override
    public void destroy() {
    }

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        //处理response的字符编码
        HttpServletResponse myResponse=(HttpServletResponse) response;
        myResponse.setContentType("text/html;charset=UTF-8");

        // 转型为与协议相关对象
        HttpServletRequest httpServletRequest = (HttpServletRequest) request;
        // 对request包装增强
        HttpServletRequest myrequest = new MyRequest(httpServletRequest);
        chain.doFilter(myrequest, response);
    }

    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
    }

}

//自定义request对象,HttpServletRequest的包装类
class MyRequest extends HttpServletRequestWrapper {

    private HttpServletRequest request;
    //是否编码的标记
    private boolean hasEncode;
    //定义一个可以传入HttpServletRequest对象的构造函数,以便对其进行装饰
    public MyRequest(HttpServletRequest request) {
        super(request);// super必须写
        this.request = request;
    }

    // 对需要增强方法 进行覆盖
    @Override
    public Map getParameterMap() {
        // 先获得请求方式
        String method = request.getMethod();
        if (method.equalsIgnoreCase("post")) {
            // post请求
            try {
                // 处理post乱码
                request.setCharacterEncoding("utf-8");
                return request.getParameterMap();
            } catch (UnsupportedEncodingException e) {
                e.printStackTrace();
            }
        } else if (method.equalsIgnoreCase("get")) {
            // get请求
            Map<String, String[]> parameterMap = request.getParameterMap();
            if (!hasEncode) { // 确保get手动编码逻辑只运行一次
                for (String parameterName : parameterMap.keySet()) {
                    String[] values = parameterMap.get(parameterName);
                    if (values != null) {
                        for (int i = 0; i < values.length; i++) {
                            try {
                                // 处理get乱码
                                values[i] = new String(values[i]
                                        .getBytes("ISO-8859-1"), "utf-8");
                            } catch (UnsupportedEncodingException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
                hasEncode = true;
            }
            return parameterMap;
        }
        return super.getParameterMap();
    }

    //取一个值
    @Override
    public String getParameter(String name) {
        Map<String, String[]> parameterMap = getParameterMap();
        String[] values = parameterMap.get(name);
        if (values == null) {
            return null;
        }
        return values[0]; // 取回参数的第一个值
    }

    //取所有值
    @Override
    public String[] getParameterValues(String name) {
        Map<String, String[]> parameterMap = getParameterMap();
        String[] values = parameterMap.get(name);
        return values;
    }
}

然后在 web.xml 中配置这个过滤器即可。

<filter>
    <filter-name>encoding</filter-name>
    <filter-class>com.yusael.filter.GenericEncodingFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>encoding</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

Json交互

json语法

{"name": "QinJiang"}
{"age": "3"}
{"sex": "男"}
var obj = {a: 'Hello', b: 'World'}; // 这是一个对象,注意键名也是可以使用引号包裹的
var json = '{"a": "Hello", "b": "World"}'; // 这是一个 JSON 字符串,本质是一个字符串

JSON 和 JavaScript 对象互转

JSON字符串 转换为 JavaScript 对象,使用 JSON.parse() 方法:

var obj = JSON.parse('{"a": "Hello", "b": "World"}');
// 结果是 {a: 'Hello', b: 'World'}

JavaScript 对象 转换为 JSON字符串,使用 JSON.stringify() 方法:

var json = JSON.stringify({a: 'Hello', b: 'World'});
// 结果是 '{"a": "Hello", "b": "World"}'

测试代码:

// 编写一个 js对象
var user = {
    name : "yusael",
    age : 3,
    sex : "男"
};
// 将 js对象 转为 json字符串
var str = JSON.stringify(user);
console.log(str);
// 将 json字符串 转为js对象
var userObj = JSON.parse(str);
console.log(userObj);

Jackson

<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.11.0</version>
</dependency>

使用 json 注意 解决乱码问题:用 @RequestMapingproduces 指定响应体返回类型和编码。

  • @RequestMapping(value = "/json", produces = "application/json;charset=utf-8")
@Controller
public class UserController {
    // 解决乱码问题
    @RequestMapping(value = "/json", produces = "application/json;charset=utf-8")
    @ResponseBody
    public String json() throws JsonProcessingException {
        // 创建一个 jackson 的对象映射器, 用来解析数据
        ObjectMapper mapper = new ObjectMapper();
        // 创建一个对象
        User user = new User(1, "zhenyu", "男");
        // 将对象解析成 json 格式
        String str = mapper.writeValueAsString(user);
        System.out.println(str);
        // 由于@ResponseBody注解, 这里会将 str 转成 json 格式返回
        return str;
    }
}

统一乱码解决

通过 @RequestMapping(value = "/json", produces = "application/json;charset=utf-8") 解决乱码比较麻烦,如果项目中有许多请求则每一个都要添加,可以通过 Spring 配置统一指定,这样就不用每次都去处理了

我们可以在 springmvc 的配置文件上添加一段消息 StringHttpMessageConverter 转换配置:

<mvc:annotation-driven>
   <mvc:message-converters register-defaults="true">
       <bean class="org.springframework.http.converter.StringHttpMessageConverter">
           <constructor-arg value="UTF-8"/>
       </bean>
       <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
           <property name="objectMapper">
               <bean class="org.springframework.http.converter.json.Jackson2ObjectMapperFactoryBean">
                   <property name="failOnEmptyBeans" value="false"/>
               </bean>
           </property>
       </bean>
   </mvc:message-converters>
</mvc:annotation-driven>

返回 Json 字符串统一解决

在类上直接使用 @RestController,这样子,里面所有的方法都只会返回 Json 字符串了,不用再每一个都添加 @ResponseBody。在前后端分离开发中,一般都使用 @RestController,十分便捷。

@RestController // 使得所有方法都返回 Json 字符串
public class UserController {
   @RequestMapping(value = "/json")
    public String json() throws JsonProcessingException {
        // 创建一个 jackson 的对象映射器, 用来解析数据
        ObjectMapper mapper = new ObjectMapper();
        // 创建一个对象
        User user = new User(1, "zhenyu", "男");
        // 将对象解析成 json 格式
        String str = mapper.writeValueAsString(user);
        System.out.println(str);
        // 由于@RestController 注解, 这里会将 str 转成 json 格式返回;
        return str;
    }
}

Json 返回集合

@RequestMapping("/json1")
public String json2() throws JsonProcessingException {
    ObjectMapper mapper = new ObjectMapper();
    User user1 = new User(1, "Jack", "男");
    User user2 = new User(2, "Rose", "女");
    User user3 = new User(3, "Kitty", "女");
    User user4 = new User(4, "Jerry", "男");
    User user5 = new User(5, "Yusael", "男");

    List<User> users = new ArrayList<>();
    users.add(user1);
    users.add(user2);
    users.add(user3);
    users.add(user4);
    users.add(user5);

    String str = mapper.writeValueAsString(users);
    return str;
}

Json 输出时间对象

默认日期格式是 时间戳 —— 1970年1月1日到当前日期的毫秒数。

  • Jackson 默认是会把时间转成 timestamps 形式。
@RequestMapping("/json2")
public String json2() throws JsonProcessingException {
    ObjectMapper mapper = new ObjectMapper();
    Date date = new Date();
    String str = mapper.writeValueAsString(date);
    return str;
}
// 输出 1591877669906

解决方案:取消 timestamps 形式,自定义时间格式。

@RequestMapping("/json3")
public String json3() throws JsonProcessingException {
    ObjectMapper mapper = new ObjectMapper();
    // 设置不使用时间戳的格式
    mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
    // 自定义日期格式对象
    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    // 指定日期格式
    mapper.setDateFormat(sdf);

    Date date = new Date();
    String str = mapper.writeValueAsString(date);
    return str;
}
// 输出: 2020-06-11 20:16:36

抽取工具类

如果要经常使用的话,这样是比较麻烦的,我们可以将这些代码封装到一个工具类中;

public class JsonUtils{
    public static String getJson(Object object){
        return getJson(object,"yyyy-MM-dd HH:mm:ss");
    }
    public static String getJson(Object object,String dateFormat){
        ObjectMapper mapper = new ObjectMapper();
         // 不使用时间戳的方式
        mapper.configura(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS,false);
         // 自定义日期格式对象
        SimpleDateFormat sdf = new SimpleDateFormat(dateFormat);
             // 指定日期格式
        mapper.setDateFormat(sdf);
        Date date = new Date();
        try {
            return mapper.writeValueAsString(date);
        } catch (JsonProcessingException e) {
            e.printStackTrace();
        }
        return null;
        
    }
}

使用工具类,代码就变得更加简单了。

@RequestMapping("/json4")
public String json4() throws JsonProcessingException {
    ObjectMapper mapper = new ObjectMapper();
    return JsonUtils.getJson(new Date());
}

fastjson

<dependency>
   <groupId>com.alibaba</groupId>
   <artifactId>fastjson2</artifactId>
   <version>1.2.60</version>
</dependency>

fastjson 中三个主要的类:

  • JSONObject 代表 Json 对象
    JSONObject 实现了 Map 接口, 猜想 JSONObject 底层操作是由 Map 实现的。
    JSONObject 对应 Json 对象,通过各种形式的 get() 方法可以获取 Json 对象中的数据,也可利用诸如 size(),isEmpty() 等方法获取 “键 : 值对” 的个数和判断是否为空。其本质是通过实现 Map 接口并调用接口中的方法完成的。

  • JSONArray 代表 Json 对象数组
    内部是由 List 接口中的方法来完成操作的。

  • JSON 代表 JSONObject 和 JSONArray 的转化

测试代码:

public class FastJsonDemo {
    public static void main(String[] args) {
        User user1 = new User(1, "Jack", "男");
        User user2 = new User(2, "Rose", "女");
        User user3 = new User(3, "Kitty", "女");
        User user4 = new User(4, "Jerry", "男");
        User user5 = new User(5, "Yusael", "男");
        List<User> list = new ArrayList<User>();
        list.add(user1);
        list.add(user2);
        list.add(user3);
        list.add(user4);
        list.add(user5);

        System.out.println("*******Java对象 转 JSON字符串*******");
        String str1 = JSON.toJSONString(list);
        System.out.println("JSON.toJSONString(list) ==> " + str1);
        String str2 = JSON.toJSONString(user1);
        System.out.println("JSON.toJSONString(user1) ==> " + str2);

        System.out.println("\n****** JSON字符串 转 Java对象*******");
        User jp_user1=JSON.parseObject(str2, User.class);
        System.out.println("JSON.parseObject(str2,User.class) ==> " + jp_user1);

        System.out.println("\n****** Java对象 转 JSON对象 ******");
        JSONObject jsonObject1 = (JSONObject) JSON.toJSON(user2);
        System.out.println("(JSONObject) JSON.toJSON(user2) ==> " + jsonObject1.getString("name"));

        System.out.println("\n****** JSON对象 转 Java对象 ******");
        User to_java_user = JSON.toJavaObject(jsonObject1, User.class);
        System.out.println("JSON.toJavaObject(jsonObject1, User.class) ==> " + to_java_user);
    }
    
}

maven

<build>
   <resources>
       <resource>
           <directory>src/main/java</directory>
           <includes>
               <include>**/*.properties</include>
               <include>**/*.xml</include>
           </includes>
           <filtering>false</filtering>
       </resource>
       <resource>
           <directory>src/main/resources</directory>
           <includes>
               <include>**/*.properties</include>
               <include>**/*.xml</include>
           </includes>
           <filtering>false</filtering>
       </resource>
   </resources>
</build>

拦截器 + 文件上传下载

拦截器(interceptor)

SpringMVC 的处理器拦截器类似于 Servlet 开发中的过滤器 Filter,用于对处理器进行 预处理 和 后处理。开发者可以自己定义一些拦截器来实现特定的功能。

过滤器与拦截器的区别:拦截器是 AOP 思想的具体应用。

过滤器(filter):

  • servlet 规范中的一部分,任何 java web 工程都可以使用;
  • 在 中配置了 /* 之后,可以对所有要访问的资源进行拦截;

拦截器(interceptor):

  • 拦截器属于 SpringMVC 框架的,只有使用了 SpringMVC 框架的工程才能使用;
  • 拦截器只会拦截访问的控制器方法, 如果访问的是 jsp/html/css/image/js 是不会进行拦截的;

自定义拦截器

想要自定义拦截器,必须实现 HandlerInterceptor 接口。

  1. 配置 web.xml 和 springmvc-servlet.xml 文件

  2. 编写一个拦截器:

    package com.yusael.interceptor;
    
    import org.springframework.web.servlet.HandlerInterceptor;
    import org.springframework.web.servlet.ModelAndView;
    
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    public class MyInterceptor implements HandlerInterceptor {
        // 在请求处理的方法之前执行
        // 如果返回 true 执行下一个拦截器(放行)
        // 如果返回 false 就不执行下一个拦截器
        @Override
        public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
            System.out.println("---------处理前---------");
            return true;
        }
    
        // 在请求处理方法执行之后执行
        @Override
        public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
            System.out.println("---------处理后---------");
        }
    
        // 在 dispatcherServlet 处理后执行, 做清理工作
        @Override
        public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
            System.out.println("---------清理---------");
        }
    
    }
    
    
  3. 在 SpringMVC 的配置文件中配置拦截器

    <!--关于拦截器的配置-->
    <mvc:interceptors>
        <mvc:interceptor>
            <!--/** 包括路径及其子路径-->
            <!--/admin/* 拦截的是/admin/add等等这种 , /admin/add/user不会被拦截-->
            <!--/admin/** 拦截的是/admin/下的所有-->
            <mvc:mapping path="/**"/>
            <!--bean配置的就是拦截器-->
            <bean class="com.yusael.interceptor.MyInterceptor"/>
        </mvc:interceptor>
    </mvc:interceptors>
    
    
  4. 编写一个 Controller,接收请求:

  5. 前端 index.jsp

    <a href="${pageContext.request.contextPath}/interceptor">拦截器测试</a>
    
  6. 启动 Tomcat 进行测试。

如果自定义拦截器中 preHandle 返回 false,表示不放行,被拦截了,不继续执行了。

在这里插入图片描述
如果自定义拦截器中 preHandle 返回 true,表示放行,继续执行后面的代码。

在这里插入图片描述

验证用户是否登录 (认证用户)

思路:

  • 有一个登陆页面,需要写一个 Controller 访问页面。
  • 登陆页面有一个提交表单的动作,需要在 Controller 中处理;
    判断用户名密码是否正确:如果正确,向 session 中写入用户信息,返回登陆成功。
  • 拦截用户请求,判断用户是否登陆。
    如果用户已经登陆,放行;
    如果用户未登陆,跳转到登陆页面;
没有拦截器

如果没有拦截器,也就是上面思路中的 1 和 2 步,用户未登录也可以进入主页

  1. 编写一个登陆页面 login.jsp

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
    <head>
        <title>login</title>
    </head>
    
    <h1>登录页面</h1>
    <hr>
    
    <body>
    <form action="${pageContext.request.contextPath}/user/login">
        用户名:<input type="text" name="username"> <br>
        密码:<input type="password" name="pwd"> <br>
        <input type="submit" value="提交">
    </form>
    </body>
    </html>
    
    
  2. 编写一个 Controller 处理请求

    package com.yusael.controller;
    
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    
    import javax.servlet.http.HttpSession;
    
    @Controller
    @RequestMapping("/user")
    public class UserController {
    
        // 跳转到登录界面
        @RequestMapping("/toLogin")
        public String toLogin() {
            return "login";
        }
    
        // 跳转到成功页面
        @RequestMapping("/toSuccess")
        public String toSucess() {
            return "success";
        }
    
        // 登录提交
        @RequestMapping("/login")
        public String login(HttpSession session, String username, String password) {
            // 向 session 记录用户身份信息
            System.out.println("接收前端 ===> " + username);
            session.setAttribute("user", username);
            return "success";
        }
    
        // 登录过期
        @RequestMapping("/logout")
        public String logout(HttpSession httpSession) {
            // session 过期
            httpSession.invalidate();
            return "login";
        }
    
    }
    
    
  3. 编写一个登陆成功的页面 success.jsp

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
    <head>
        <title>success</title>
    </head>
    <body>
    
    <h1>登录成功页面</h1>
    <hr>
    
    ${user}
    <a href="${pageContext.request.contextPath}/user/logout">注销</a>
    </body>
    </html>
    
    
  4. 编写主页 index.jsp

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
    <head>
        <title>index</title>
    </head>
    <body>
    <h1>首页</h1>
    <hr>
    <%--登录--%>
    <a href="${pageContext.request.contextPath}/user/toLogin">登录</a>
    <a href="${pageContext.request.contextPath}/user/toSuccess">成功页面</a>
    </body>
    </html>
    
    
配置拦截器
  1. 编写用户登录拦截器 LoginInterceptor.java

    package com.yusael.interceptor;
    
    import org.springframework.web.servlet.HandlerInterceptor;
    import org.springframework.web.servlet.ModelAndView;
    
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    import javax.servlet.http.HttpSession;
    
    public class LoginInterceptor implements HandlerInterceptor{
       @Override
        public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
            // 如果是登录页面则放行
            System.out.println(request.getRequestURI());
            if(request.getRequestURI().contains("login")){
                return true;
            }
            HttpSession session = request.getSession();
            // 如果用户已经登陆也放行
            if(session.getAttribute("user") != null){
                return true;
            }
            // 如果用户没有登录则跳转到登录界面
            request.getRequestDispatcher("/WEB-INF/jsp/login.jsp").forward(request,response);
            return false;
        } 
        @Override
        public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
    
        }
    
        @Override
        public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
    
        }
    
    }
    
  2. 在 Springmvc 的配置文件中注册拦截器:

    <!--关于拦截器的配置-->
    <mvc:interceptors>
        <mvc:interceptor>
            <mvc:mapping path="/**"/>
            <bean id="loginInterceptor" class="com.yusael.interceptor.LoginInterceptor"/>
        </mvc:interceptor>
    </mvc:interceptors>
    
    

    由于配置了 拦截器,直接登录 成功页面,虽然会请求跳转到 success.jsp 页面,但是由于不满足拦截器进行了判断,不满足放行的条件则会跳转到 登录界面。只有满足拦截器放行的条件才会跳转到 成功页面

文件的上传

​ 文件上传是项目开发中最常见的功能之一 ,SpringMVC 可以很好的支持文件上传,但是 SpringMVC 上下文中默认没有装配 MultipartResolver,因此默认情况下其不能处理文件上传工作。如果想使用Spring 的文件上传功能,则需要在上下文中配置 MultipartResolver。

对前端表单的要求:为了能上传文件,必须将表单的 method 设置为 POST,并将 enctype 设置为multipart/form-data。只有在这样的情况下,浏览器才会把用户选择的文件以二进制数据发送给服务器;

表单中的 enctype (编码方式)属性的说明:

  • application/x-www=form-urlencoded:默认方式,只处理表单域中的 value 属性值,采用这种编码方式的表单会将表单域中的值处理成 URL 编码方式。
  • multipart/form-data:这种编码方式会以二进制流的方式来处理表单数据,这种编码方式会把文件域指定文件的内容也封装到请求参数中,不会对字符编码。
  • text/plain:除了把空格转换为 “+” 号外,其他字符都不做编码处理,这种方式适用直接通过表单发送邮件。
<form action="" enctype="multipart/form-data" method="post">
   <input type="file" name="file"/>
   <input type="submit">
</form>

一旦设置了 enctypemultipart/form-data,浏览器即会采用二进制流的方式来处理表单数据,而对于文件上传的处理则涉及在服务器端解析原始的 HTTP 响应。

在2003年,Apache Software Foundation 发布了开源的 Commons FileUpload 组件,其很快成为Servlet/JSP 程序员上传文件的最佳选择。

Servlet3.0 规范已经提供方法来处理文件上传,但这种上传需要在 Servlet 中完成;而Spring MVC则提供了更简单的封装;

Spring MVC 为文件上传提供了直接的支持,这种支持是用即插即用的 MultipartResolver 实现的。

Spring MVC 使用 Apache Commons FileUpload 技术实现了一个 MultipartResolver 实现类:
CommonsMultipartResolver。因此,SpringMVC 的文件上传还需要依赖 Apache Commons FileUpload 的组件。

文件输出流进行文件上传
  1. 引入 commons-fileupload 的依赖

    <!--文件上传-->
    <dependency>
       <groupId>commons-fileupload</groupId>
       <artifactId>commons-fileupload</artifactId>
       <version>1.3.3</version>
    </dependency>
    <!--servlet-api导入高版本的-->
    <dependency>
       <groupId>javax.servlet</groupId>
       <artifactId>javax.servlet-api</artifactId>
       <version>4.0.1</version>
    </dependency>
    
    
  2. 配置 bean:multipartResolver

    注:这个 bean 的 id 必须为:multipartResolver,否则上传文件会报 400 的错误。

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

    CommonsMultipartFile 的 常用方法:

    String getOriginalFilename() 获取上传文件的原名
    InputStream getInputStream() 获取文件流
    void transferTo(File dest)   将上传文件保存到一个目录文件中
    
  3. 编写前端页面:

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
      <head>
        <title>file</title>
      </head>
      <body>
        <form action="/upload" enctype="multipart/form-data" method="post">
          <input type="file" name="file"/>
          <input type="submit" value="upload">
        </form>
      </body>
    </html>
    
    
  4. Controller

    package com.yusael.controller;
    
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.multipart.commons.CommonsMultipartFile;
    
    import javax.servlet.http.HttpServletRequest;
    import java.io.*;
    
    @Controller
    public class FileController {
        // @RequestParam("file") 将 name=file 控件得到的文件封装成 CommonsMultipartFile 对象
        // 批量上传 CommonsMultipartFile 则为数组即可
        @RequestMapping("/upload")
        public String fileUpload(@RequestParam("file")CommonsMultipartFile file, HttpServletRequest request) throws IOException {
            // 获取文件名
            String uploadFileName = file.getOriginalFilename();
    
            // 如果文件名为空, 直接回到首页
            if ("".equals(uploadFileName)){
                return "redirect:/index.jsp";
            }
    
            System.out.println("上传文件名: " + uploadFileName);
    
            // 上传路径保存设置
            String path = request.getServletContext().getRealPath("/upload");
            // 如果路径不存在, 则创建一个
            File realPath = new File(path);
            if (!realPath.exists()) {
                realPath.mkdirs();
            }
            System.out.println("上传文件保存地址: " + realPath);
    
            // 文件输入流
            InputStream is = file.getInputStream();
            // 文件输出流
            OutputStream os = new FileOutputStream(new File(realPath, uploadFileName));
    
            // 读取写出
            int len = 0;
            byte[] buffer = new byte[1024];
            while ((len = is.read(buffer)) != -1) {
                os.write(buffer, 0, len);
                os.flush();
            }
            os.close();
            is.close();
            return "redirect:/index.jsp";
        }
    
    }
    
    
file.Transto 进行文件上传

Controller:

@RequestMapping("/upload2")
public String fileUpload2(@RequestParam("file") CommonsMultipartFile file, HttpServletRequest request) throws IOException {
    // 上传路径保存设置
    String path = request.getServletContext().getRealPath("/upload");
    File realPath = new File(path);
    if (!realPath.exists()) {
        realPath.mkdir();
    }
    System.out.println("上传文件保存地址: " + realPath);

    // 通过 CommonsMultipartFile 的方法直接写文件
    file.transferTo(new File(realPath + "/" + file.getOriginalFilename()));

    return "redirect:/index.jsp";
}

前端页面 index.jsp:

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
  <head>
    <title>file</title>
  </head>
  <body>
    <form action="/upload" enctype="multipart/form-data" method="post">
      <input type="file" name="file"/>
      <input type="submit" value="upload">
    </form>
    <hr>
    <form action="/upload2" enctype="multipart/form-data" method="post">
      <input type="file" name="file"/>
      <input type="submit" value="upload">
    </form>
  </body>
</html>

文件的下载

文件下载步骤:

  1. 设置 response 响应头
  2. 读取文件 – InputStream
  3. 写出文件 – OutputStream
  4. 执行操作
  5. 关闭流 (先开后关)

代码实现:

@RequestMapping("/download")
public String downloads(HttpServletResponse response , HttpServletRequest request) throws Exception{
    // 要下载的图片地址
    String  path = request.getServletContext().getRealPath("/upload");
    String  fileName = "SpringBoot笔记.pdf";

    // 1、设置response 响应头
    response.reset(); // 设置页面不缓存, 清空buffer
    response.setCharacterEncoding("UTF-8"); // 字符编码
    response.setContentType("multipart/form-data"); // 二进制传输数据
    // 设置响应头
    response.setHeader("Content-Disposition",
            "attachment;fileName="+ URLEncoder.encode(fileName, "UTF-8"));

    File file = new File(path,fileName);

    // 2、读取文件--输入流
    InputStream input = new FileInputStream(file);

    // 3、写出文件--输出流
    OutputStream out = response.getOutputStream();

    byte[] buff = new byte[1024];
    int len = 0;

    // 4、执行 写出操作
    while ((len = input.read(buff)) != -1) {
        out.write(buff, 0, len);
        out.flush();
    }
    out.close();
    input.close();
    return null;
}

前端页面 index.jsp:

<a href="/download">点击下载</a>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值