最近做项目需要使用切面技术,对已有系统使用的组件进行切面拦截入参和出参,梳理下不同切面技术的使用和差别。
Spring Aop
特点
- 动态代理的方式实现Aop,接口类型通过JDK代理实现,非接口类型通过cglib代理实现
- 只能切面Spring管理的bean
- 运行时织入
- 针对bean自身的接口无法使用代理,故无法进行切面
使用方式
基于Springboot,调用DB Service接口时进行代理配置,这样就可以做到根据不同的环境配置不同的代理
@SpringBootApplication
@Configuration
@ComponentScan
@EnableAutoConfiguration
@EnableScheduling
@MapperScan("com.xx.ci.dtw.dto.dao")
public class Application {
public static void main(String[] args) throws Exception {
SpringApplication.run(Application.class, args);
}
}
切面实现
@Component
public class DBPrxoyAspect {
private final static Logger logger = LoggerFactory.getLogger(DBPrxoyAspect.class);
@Value("${socksProxyHost}")
private String socksProxyHost;
@Value("${socksProxyPort}")
private String socksProxyPort;
@Pointcut("execution(public * com.xx.ci.dtw.service..*.*(..))")
public void serviceDbAop() {
}
@Before("serviceDbAop()")
public void doBefore(JoinPoint joinPoint) throws Throwable {
Properties prop = System.getProperties();
prop.setProperty("socksProxyHost", socksProxyHost);
prop.setProperty("socksProxyPort", socksProxyPort);
logger.debug("add db proxy, socksProxyHost={}, socksProxyPort={}", socksProxyHost, socksProxyPort);
}
@org.aspectj.lang.annotation.After("serviceDbAop()")
public void After(JoinPoint joinPoint) throws Throwable {
Properties prop = System.getProperties();
prop.setProperty("socksProxyHost", "");
prop.setProperty("socksProxyPort", "");
logger.debug("release db proxy, socksProxyHost={}, socksProxyPort={}", socksProxyHost, socksProxyPort);
}
}
AspectJ
特点
Spring Aop 很多地方都是直接用到AspectJ里面的代码。典型的比如@Aspect,@Around,@Pointcut注解等等。而且从相关概念以及语法结构上而言,两者其实非常非常相似
最大的区别在于两者实现AOP的底层原理不太一样:
Spring AOP: 基于代理(Proxying)
AspectJ: 基于字节码操作(Bytecode Manipulation)
使用方式
参考:https://github.com/medvedev1088/aspectj-load-time-weaving-example
- 实现切面功能
@Aspect
public class DateTimeToStringAspect {
public static final String TO_STRING_RESULT = "test";
@Pointcut("execution(* org.joda.time.base.AbstractDateTime.toString())")
public void dateTimeToString() {
}
@Around("dateTimeToString()")
public Object toLowerCase(ProceedingJoinPoint joinPoint) throws Throwable {
Object ignoredToStringResult = joinPoint.proceed();
System.out.println("DateTime#toString() has been invoked: " + ignoredToStringResult);
return TO_STRING_RESULT;
}
}
2.配置切面和切点信息
<aspectj>
<aspects>
<!-- Aspects -->
<aspect name="com.example.aspectj.DateTimeToStringAspect"/>
</aspects>
<weaver options="-verbose -showWeaveInfo">
<include within="org.joda.time.base.AbstractDateTime"/>
</weaver>
</aspectj>
3、使用javaagent实现classloader加载时织入
启动运行时增加vm的代理配置
若是使用Idea 直接run 则需要配置Edit Configration 增加VM配置:
-javaagent:/Users/sherrichen/.m2/repository/org/aspectj/aspectjweaver/1.8.13/aspectjweaver-1.8.13.jar
jar的运行方式增加代理
java -javaagent:/Users/sherrichen/.m2/repository/org/aspectj/aspectjweaver//1.8.13/aspectjweaver-1.8.13.jar -jar target/aspectj-ltw-example-1.0-SNAPSHOT.jar