1. 程式人生 > 其它 >acwing 786. 第k個數

acwing 786. 第k個數

攔截器

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

自定義攔截器實現步驟

攔截器的執行

自定義攔截器,需要實現 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 方法中加入資料。

攔截器中方法與處理器方法的執行順序如下圖:

多個攔截器執行

當有多個攔截器時,形成攔截器鏈。攔截器鏈的執行順序,與其註冊順序一致。需要再次強調一點的是,當某一個攔截器的 preHandle()方法返回 true 並被執行到時,會向一個專門的方法棧中放入該攔截器的 afterCompletion()方法。

多個攔截器中方法與處理器方法的執行順序如下圖:

從圖中可以看出,只要有一個 preHandle()方法返回 false,則上部的執行鏈將被斷開,其後續的處理器方法與 postHandle()方法將無法執行。但,無論執行鏈執行情況怎樣,只要方法棧中有方法,即執行鏈中只要有 preHandle()方法返回 true,就會執行方法棧中的afterCompletion()方法。最終都會給出響應。

攔截器登入驗證例子

  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>