天天看點

SpringMVC攔截器

攔截器

SpringMVC 中的 Interceptor 攔截器是非常重要和相當有用的,它的主要作用是攔截指定的使用者請求,并進行相應的預處理與後處理。其攔截的時間點在“處理器映射器根據使用者送出的請求映射出了所要執行的處理器類,并且也找到了要執行該處理器類的處理器擴充卡,在處理器擴充卡執行處理器之前”。當然,在處理器映射器映射出所要執行的處理器類時,已經将攔截器與處理器組合為了一個處理器執行鍊,并傳回給了中央排程器。

自定義攔截器實作步驟

SpringMVC攔截器

攔截器的執行

自定義攔截器,需要實作 HandlerInterceptor 接口。而該接口中含有三個方法:

public class MyInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        return HandlerInterceptor.super.preHandle(request, response, handler);
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        HandlerInterceptor.super.postHandle(request, response, handler, modelAndView);
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        HandlerInterceptor.super.afterCompletion(request, response, handler, ex);
    }
}
           

preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)

該方法在處理器方法執行之前執行。其傳回值為 boolean,若為 true,則緊接着會執行處理器方法,且會将 afterCompletion()方法放入到一個專門的方法棧中等待執行。

postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView)

該方法在處理器方法執行之後執行。處理器方法若最終未被執行,則該方法不會執行。由于該方法是在處理器方法執行完後執行,且該方法參數中包含ModelAndView,是以該方法可以修改處理器方法的處理結果資料,且可以修改跳轉方向。

afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)

當 preHandle()方法傳回 true 時,會将該方法放到專門的方法棧中,等到對請求進行響應的所有工作完成之後才執行該方法。即該方法是在中央排程器渲染(資料填充)了響應頁面之後執行的,此時對 ModelAndView 再操作也對響應無濟于事。

afterCompletion 最後執行的方法,清除資源,例如在 Controller 方法中加入資料。

多個攔截器執行

攔截器登入驗證例子

  1. 建立controller,doLogin方法不進行攔截,doSome方法進行攔截。doLogin方法進行驗證登入資訊。
@Controller
@RequestMapping("/user")
public class UserController {
    @RequestMapping("/login.do")
    public ModelAndView doLogin(HttpSession session, String name, Integer age){
        ModelAndView modelAndView = new ModelAndView();
        modelAndView.addObject("uname",name);
        modelAndView.addObject("uage",age);
        modelAndView.setViewName("forward:/user/some.do");
        if(name.equals("黃振聰")) {
            session.setAttribute("uname",name);
        }
        return modelAndView;
    }
    @RequestMapping("/some.do")
    public String doSome(){
        return "target";
    }
}
           
  1. 建立攔截器實作類。
public class UserInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("執行preHandle...");
        HttpSession session = request.getSession();
        String uname = (String) session.getAttribute("uname");
        if(uname != null){
            return true;
        }
        return false;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("執行postHandle...");
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("執行afterCompletion...");
    }
}

           
  1. 在主配置檔案中聲明攔截器
<mvc:interceptors>
        <mvc:interceptor>
            <mvc:mapping path="/user/some.do"/>
            <bean class="com.hzc.interceptor.UserInterceptor" />
        </mvc:interceptor>
    </mvc:interceptors>
           
  1. 建立登入界面與結果界面
<body>
    <form action="user/login.do">
        姓名:<input type="text" name="name"><br>
        年齡:<input type="text" name="age"><br>
        <input type="submit" value="登入">
    </form>
</body>



<body>
    <h2>姓名:${requestScope.uname}</h2>
    <h2>年齡:${requestScope.uage}</h2>
</body>