SpringMVC杂记(十五) spring-mvc controller 的切面

SpringMVC杂记(十五)spring-mvccontroller的切面

一)对一般Spring上下文管理的bean,配合切面表达式都可以对其做拦截操作,原理是Spring在启动的时候,利用字节码技术生成了其子类,

这样就把切面逻辑“织入”到bean之中了。这个用spring的人一般都懂。但是很多人发现,spring-mvc使用起来,controller的代码不能通过这种

方式自己织入逻辑。

SpringMVC在启动的时候会根据处理器策略加载handler映射到一个map中,

这个时候因为配置的原因它从容器中取到的对象是原生对象,而不是我们代理的对象;所以不管我们的Controller做了什么操作都不会经过我们的代理,这样AOP就失去了作用。

二)有些情况,确实需要做代理,而且Interceptor有不太好使的时候,我们就要另辟蹊径了。如下代码和配置所示,一样还是可以成功的。缺点就是

切面表达式不好用了,要自己写代码判断方法是不是要拦截。

<bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" depends-on="lifecycleBeanPostProcessor" />
import java.lang.reflect.Method;

import org.aopalliance.aop.Advice;
import org.aopalliance.intercept.MethodInvocation;
import org.springframework.aop.support.StaticMethodMatcherPointcutAdvisor;
import org.springframework.beans.factory.InitializingBean;


@SuppressWarnings({"serial"})
public class SpringMvcControllerAspect extends StaticMethodMatcherPointcutAdvisor implements InitializingBean {

//	private static final Logger LOGGER = LoggerFactory.getLogger(SpringMvcControllerAspect.class);
	
	@Override
	public boolean matches(Method method, Class<?> targetClass) {
		// 这里来判断方法是否需要拦截
		// 不用切点表达式,自己写代码判断
		return false;
	}
	
	@Override
	public void afterPropertiesSet() throws Exception {
		super.setOrder(Integer.MAX_VALUE);
		super.setAdvice(MethodInterceptor.INSTANCE);
	}

	// 织入业务逻辑
	// ================================================================================================================
	private static final class MethodInterceptor implements org.aopalliance.intercept.MethodInterceptor, Advice {
		public static final Advice INSTANCE = new MethodInterceptor();
		
		public Object invoke(MethodInvocation invocation) throws Throwable {
			// 前织入
			Object result = invocation.proceed();
			// 后织入
			return result;
		}
	}
}

相关推荐