Spring MVC 中 HandlerInterceptorAdapter 的使用

发布于 2024-07-14 07:36:53 字数 4475 浏览 24 评论 0

一般情况下,对来自浏览器的请求的拦截,是利用 Filter 实现的,这种方式可以实现 Bean 预处理、后处理。Spring MVC 的拦截器不仅可实现 Filter 的所有功能,还可以更精确的控制拦截精度。

spring 为我们提供了 org.springframework.web.servlet.handler.HandlerInterceptorAdapter 这个适配器,继承此类,可以非常方便的实现自己的拦截器。他有三个方法:

package org.springframework.web.servlet.handler;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.web.servlet.AsyncHandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;

public abstract class HandlerInterceptorAdapter implements AsyncHandlerInterceptor {

	public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
		throws Exception {
		return true;
	}

	public void postHandle(
			HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView)
			throws Exception {
	}

	public void afterCompletion(
			HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
			throws Exception {
	}

	public void afterConcurrentHandlingStarted(
			HttpServletRequest request, HttpServletResponse response, Object handler)
			throws Exception {
	}

}

其中 afterConcurrentHandlingStarted 方法是其父类的

这三个方法分别实现预处理、后处理(调用了 Service 并返回 ModelAndView,但未进行页面渲染)、返回处理(已经渲染了页面)
在 preHandle 中,可以进行编码、安全控制等处理; 在 postHandle 中,有机会修改 ModelAndView; 在 afterCompletion 中,可以根据 ex 是否为 null 判断是否发生了异常,进行日志记录。

如果基于 xml 配置使用 Spring MVC,可以利用 SimpleUrlHandlerMapping、BeanNameUrlHandlerMapping 进行 Url 映射(相当于 struts 的 path 映射)和拦截请求(注入 interceptors),如果基于注解使用 Spring MVC,可以使用 DefaultAnnotationHandlerMapping 注入 interceptors。

注意无论基于 xml 还是基于注解,HandlerMapping bean 都是需要在 xml 中配置的。

一个 demo:在这个例子中,我们假设 UserController 中的注册操作只在 9:00-12:00 开放,那么就可以使用拦截器实现这个功能。

public class TimeBasedAccessInterceptor extends HandlerInterceptorAdapter {    
    private int openingTime;    
    private int closingTime;    
    private String mappingURL;//利用正则映射到需要拦截的路径    
    public void setOpeningTime(int openingTime) {    
        this.openingTime = openingTime;    
    }    
    public void setClosingTime(int closingTime) {    
        this.closingTime = closingTime;    
    }    
    public void setMappingURL(String mappingURL) {    
        this.mappingURL = mappingURL;    
    }    
    @Override    
    public boolean preHandle(HttpServletRequest request,    
            HttpServletResponse response, Object handler) throws Exception {    
        String url=request.getRequestURL().toString();    
        if(mappingURL==null || url.matches(mappingURL)){    
            Calendar c=Calendar.getInstance();    
            c.setTime(new Date());    
            int now=c.get(Calendar.HOUR_OF_DAY);    
            if(now<openingTime || now>closingTime){    
                request.setAttribute("msg", "注册开放时间:9:00-12:00");    
                request.getRequestDispatcher("/msg.jsp").forward(request, response);    
                return false;    
            }    
            return true;    
        }    
        return true;    
    }    
}    

xml 配置:

<bean id="timeBasedAccessInterceptor" class="com.spring.handler.TimeBasedAccessInterceptor">    
    <property name="openingTime" value="9" />    
    <property name="closingTime" value="12" />    
    <property name="mappingURL" value=".*/user\.do\?action=reg.*" />    
</bean>    
<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping">    
    <property name="interceptors">    
        <list>    
            <ref bean="timeBasedAccessInterceptor"/>    
        </list>    
    </property>    
</bean>    

这里我们定义了一个 mappingURL 属性,实现利用正则表达式对 url 进行匹配,从而更细粒度的进行拦截。当然如果不定义 mappingURL,则默认拦截所有对 Controller 的请求。

UserController:

@Controller    
@RequestMapping("/user.do")    
public class UserController{    
    @Autowired    
    private UserService userService;    
    @RequestMapping(params="action=reg")    
    public ModelAndView reg(Users user) throws Exception {    
        userService.addUser(user);    
        return new ModelAndView("profile","user",user);    
    }    
    // other option ...    
}  

这个 Controller 相当于 Struts 的 DispatchAction,你也可以配置多个拦截器,每个拦截器进行不同的分工。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

文章
评论
28 人气
更多

推荐作者

櫻之舞

文章 0 评论 0

弥枳

文章 0 评论 0

m2429

文章 0 评论 0

野却迷人

文章 0 评论 0

我怀念的。

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文