Java 杂谈Spring-Boot

Spring Aop和AspectJ 面向切面技术

2019-01-08  本文已影响22人  小雪的笔记

最近做项目需要使用切面技术,对已有系统使用的组件进行切面拦截入参和出参,梳理下不同切面技术的使用和差别。

Spring Aop

特点
使用方式

基于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

  1. 实现切面功能
@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

Spring Aop和AspectJ区别(参考网上资料)

https://juejin.im/post/5a695b3cf265da3e47449471#comment

上一篇下一篇

猜你喜欢

热点阅读