Spring Core – Phần 5: Spring AOP là gì? code ví dụ với Spring AOP

Spring Core – Phần 5: Spring AOP là gì? code ví dụ với Spring AOP

1. Spring AOP là gì?

Aspect Oriented Programming (AOP) là 1 kỹ thuật lập trình dùng để tách logic chương trình thành các phần riêng biệt…

Trong Spring AOP, có 4 loại advice được hỗ trợ:

  • Before advice: chạy trước khi method được thực thi
  • After returning advice: Chạy sau khi method trả về một kết quả
  • After throwing adivce: Chạy khi method ném ra một exception
  • Around advice: Chạy khi method được thực thi (Bao gồm cả 3 loại advice trên)

 

Ở bài trước mình đã giới thiệu rõ AOP là gì, tác dụng và đặc điểm của nó nên trong bài này mình sẽ chủ yếu tập trung vào cách thực thi AOP trong Spring.

2. Code ví dụ với Spring AOP.

Nhắc lại một chút về một số khái nhiệm và thuật ngữ trong AOP với Spring (Phần này hơi trừu tượng một chút, các bạn đối chiếu với ví dụ bên dưới để hiểu hơn)

  • Join point: là các điểm trong chương trình ví dụ điểm thực thi method (method execution), điểm xử lý excpetion, field access… Spring chỉ hỗ trợ method execution join point
  • Advice: một hành động thực hiện ở joint point
  • Pointcut: Là expression language giúp khớp nối với joint point
  • Introduction: Cho phép introduce các new interface tới bất kì object adviced nào.
  • Target Object: Object sẽ được adviced
  • Aspect: là một class bao gồm các advice và các joint point
  • Interceptor: là một aspect chỉ có duy nhất một advice
  • AOP Proxy: dùng để cài đặt các Aspect
  • Weaving: là tiến trình nối các aspect với các object, types để tạo nên advised object.

Spring Core - Phần 5: Spring AOP là gì? code ví dụ với Spring AOP

Ví dụ kinh điểm với AOP mà ta hay dùng đó là chương trình thực hiện log.

Bây giờ mình sẽ tạo 1 class với các method, sau đó áp dụng AOP để thực hiện log các method của class theo cả 4 loại advice:

Khai báo các thư viện để sử dụng applicationContext, bean, aop trong spring:

<!-- https://mvnrepository.com/artifact/org.springframework/spring-beans -->
<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-beans</artifactId>
  <version>4.3.13.RELEASE</version>
</dependency>

<!-- https://mvnrepository.com/artifact/org.springframework/spring-context -->
<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-context</artifactId>
  <version>4.3.13.RELEASE</version>
</dependency>

<!-- https://mvnrepository.com/artifact/org.springframework/spring-aop -->
<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-aop</artifactId>
  <version>4.3.13.RELEASE</version>
</dependency>

Tạo class Hello.java với 3 method:

public class Hello {
  public void method1() {
    System.out.println("+++++++++++++++++++++++++++++++");
    System.out.println("method 1");
  }

  public void method2() {
    System.out.println("+++++++++++++++++++++++++++++++");
    System.out.println("method 2");
  }

  public void method3() {
    System.out.println("+++++++++++++++++++++++++++++++");
    System.out.println("method 3");
    throw new IllegalArgumentException();
  }
}

applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
  xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">

  <bean id="hello" class="stackjava.com.springaop.Hello" />
</beans>

Demo:

public static void main(String[] args) throws Exception {
  ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
  Hello hello = (Hello) context.getBean("hello");
  hello.method1();
  hello.method2();
}

Kết quả:

+++++++++++++++++++++++++++++++
method 1
+++++++++++++++++++++++++++++++
method 2

 

Áp dụng before advice

Tạo class DemoBeforeAdvice.java định nghĩa hành động trước khi cách method của class Hello được chạy:

public class DemoBeforeAdvice implements MethodBeforeAdvice {

  public void before(Method method, Object[] args, Object target) throws Throwable {
    System.out.println("before method: " + method.getName());
  }

}

MethodBeforeAdvice: là một aspect đồng thời cũng là một Interceptor vì nó chỉ có 1 advice

DemoBeforeService: là một AOP proxy, nó cài đặt lại aspect là MethodBeforeAdvice

aplicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
  xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">

  <bean id="hello" class="stackjava.com.springaop.Hello" />
  <bean id="demoBeforeAdvice" class="stackjava.com.springaop.DemoBeforeAdvice" />

  <bean id="helloProxy" class="org.springframework.aop.framework.ProxyFactoryBean">

    <property name="target" ref="hello" />

    <property name="interceptorNames">
      <list>
        <value>demoBeforeAdvice</value>
      </list>
    </property>
  </bean>

</beans>

target object sẽ là đối tượng hello.

Demo:

public static void main(String[] args) throws Exception {
  ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
  Hello hello = (Hello) context.getBean("helloProxy");
  hello.method1();
  hello.method2();
}

Kết quả:

before method: method1
+++++++++++++++++++++++++++++++
method 1
before method: method2
+++++++++++++++++++++++++++++++
method 2

Tương tự mình sẽ tạo các proxy thực hiện cài đặt After returning advice, After throwing adivce, Around advice:

import java.lang.reflect.Method;

import org.springframework.aop.AfterReturningAdvice;

public class DemoAfterAdvice implements AfterReturningAdvice {

  public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
    System.out.println("after method: " + method.getName());
    
  }

}
import org.springframework.aop.ThrowsAdvice;

public class DemoThrowAdvice implements ThrowsAdvice {

  public void afterThrowing(IllegalArgumentException e) throws Throwable {
    System.out.println("throw advice method: " );
  }

}
import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;

public class DemoAroundAdvice implements MethodInterceptor {

  public Object invoke(MethodInvocation invocation) throws Throwable {
    // same with MethodBeforeAdvice
    System.out.println("around - before method: " + invocation.getMethod().getName());

    try {
      // proceed to original method call
      Object result = invocation.proceed();

      // same with AfterReturningAdvice
      System.out.println("around - before method: " + invocation.getMethod().getName());

      return result;

    } catch (IllegalArgumentException e) {
      // same with ThrowsAdvice
      System.out.println("around - throw advice method: " + invocation.getMethod().getName());
      throw e;
    }
  }
}

 

applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
  xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">

  <bean id="hello" class="stackjava.com.springaop.Hello" />
  <bean id="demoBeforeAdvice" class="stackjava.com.springaop.DemoBeforeAdvice" />
  <bean id="demoAfterAdvice" class="stackjava.com.springaop.DemoAfterAdvice" />
  <bean id="demoThrowAdvice" class="stackjava.com.springaop.DemoThrowAdvice" />
  <bean id="demoAroundAdvice" class="stackjava.com.springaop.DemoAroundAdvice" />

  <bean id="helloProxy" class="org.springframework.aop.framework.ProxyFactoryBean">

    <property name="target" ref="hello" />

    <property name="interceptorNames">
      <list>
        <value>demoBeforeAdvice</value>
        <value>demoAfterAdvice</value>
        <value>demoThrowAdvice</value>
        <value>demoAroundAdvice</value>
      </list>
    </property>
  </bean>

</beans>

Demo:

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MainApp {
  public static void main(String[] args) throws Exception {
    ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
    Hello hello = (Hello) context.getBean("helloProxy");
    hello.method1();
    System.out.println("\n");
    hello.method2();
    System.out.println("\n");
    hello.method3();
  }
}

Kết quả:

before method: method1
around - before method: method1
+++++++++++++++++++++++++++++++
method 1
around - before method: method1
after method: method1


before method: method2
around - before method: method2
+++++++++++++++++++++++++++++++
method 2
around - before method: method2
after method: method2


before method: method3
around - before method: method3
+++++++++++++++++++++++++++++++
method 3
around - throw advice method: method3
throw advice method: 
Exception in thread "main" java.lang.IllegalArgumentException

Okay, Done!

Các bạn có thể download code ví dụ trên tại đây

References: https://docs.spring.io/spring/docs/3.0.x/reference/aop.html

stackjava.com