Aspect Oriented Programming (AOP) merupakan paradigma pemrograman dimana kita bisa memisahkan logic tertentu secara terpusat dan menyisipkannya ke dalam objek tanpa harus mengubah objek tersebut secara langsung. Misalnya pada sebuah method yang melakukan sebuah action, kita ingin menyisipkan behavior tambahan pada method tersebut tanpa harus menulisnya pada method tersebut secara langsung. Contoh kasus yang sering menggunakan AOP meliputi setup config, caching, logging, error handling, validation, dan lainnya. Contohnya adalah anotasi @Transactional
pada Spring Data yang secara "magic" menyisipkan begin transaction, commit atau rollback tanpa perlu kita tulis secara manual. Atau anotasi @BeforeEach
dan @AfterEach
pada JUnit yang membuat kita bisa menyisipkan setup tertentu pada setiap test method saat dieksekusi tanpa harus menulisnya satu-persatu pada tiap method. Juga anotasi @ControllerAdvice
pada Spring web untuk menyisipkan logic catch exception secara terpusat. Dan masih ada beberapa contoh lainnya. Ada beberapa cara untuk mengimplementasinya, tapi yang paling umum adalah menggunakan AspectJ dan Spring AOP.
Terminology
Terdapat 3 terminology pada AOP:
- Pointcut, yaitu ekspresi atau kondisi yang menjadi penanda kapan method tersebut akan disisipkan, seperti annotasi tertentu atau pola tertentu.
- Advice, yaitu logic yang akan kita sisipkan ketika method tersebut memenuhi syarat pada Pointcut.
- Aspect, yaitu tempat konfigurasi Advice dan Pointcut tersebut.
AspectJ
Ini adalah framework native AOP pada Java. Di sini behavior tambahan tersebut disisipkan secara native ke dalam object saat compile. Jadi secara teknis behavior yang kita buat itu benar-benar disisipkan ke dalam method yang digunakan. Ketika menggunakan AspectJ kita tidak bisa menggunakan compiler standar dari Java seperti “javac”, melainkan menggunakan compiler bawaan dari AspectJ, yaitu “ajc”. Kelemahan menggunakan AspectJ adalah sering bermasalah dengan library annotation processor lainnya seperti Lombok, Mapstruct, dan sejenisnya karena tidak compatible dengan compilernya. Berikut ini contoh setup AOP menggunakan AspectJ.
AspectJ pom.xml
<dependencies>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjrt</artifactId>
<version>1.8.9</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.9</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>aspectj-maven-plugin</artifactId>
<version>1.14.0</version>
<configuration>
<complianceLevel>1.8</complianceLevel>
<source>1.8</source>
<target>1.8</target>
<showWeaveInfo>true</showWeaveInfo>
<verbose>true</verbose>
<Xlint>ignore</Xlint>
<encoding>UTF-8</encoding>
</configuration>
<executions>
<execution>
<goals>
<goal>compile</goal>
<goal>test-compile</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
Spring AOP
Cara kedua adalah menggunakan Spring AOP. Sebenarnya Spring AOP ini juga membawa beberapa konfigurasi dari AspectJ, hanya saja implementasinya sangat berbeda. Setup project-nya jauh lebih gampang. Di sini behavior tambahan itu disisipkan lewat Proxy, bukan disisipkan secara native. Untuk lebih jelasnya gimana cara kerja Proxy bisa baca Proxy Design Pattern pada tulisan sebelumnya. Bedanya di sini Spring menggunakan library untuk membuat objek Proxy agar di-generate ketika runtime, bukan dibuat manual. Keunggulannya kita tetap bisa menggunakan “javac” compiler tanpa harus setup “ajc” compiler seperti AspectJ sehingga tetap compatible dengan library annotation processor lain. Kekurangannya adalah ini hanya bisa diimplementasi pada Spring Bean saja, seperti class yang memiliki annotasi @Component
, @Service
, @Controller
, @RestController
, @Repository
, @Configuration
, atau objek yang dibuat lewat @Bean
method. Selain itu secara teori menggunakan Proxy untuk menyisipkan behavior tersebut saat runtime tentu lebih lambat dibandingkan disisipkan secara native saat compile. Berikut ini adalah contoh setup AOP menggunakan Spring AOP.
Spring AOP pom.xml
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
</dependencies>
Contoh Use Case
Misalkan kita memiliki class Calculator seperti berikut dan ingin menyisipkan beberapa behavior tambahan pada masing-masing method tanpa harus copy-paste behavior tersebut secara langsung ke dalam method.
Calculator
public class Calculator{
public int add(int x, int y){
int result = x + y;
System.out.println("x + y = " + result);
return result;
}
public int subtract(int x, int y){
int result = x - y;
System.out.println("x - y = " + result);
return result;
}
public int multiply(int x, int y){
int result = x * y;
System.out.println("x * y = " + result);
return result;
}
public int divide(int x, int y){
int result = x / y;
System.out.println("x / y = " + result);
return result;
}
}
Untuk yang menggunakan Spring AOP, perlu tambahkan annotasi @Component
pada class Calculator biar jadi Spring Bean. Kita juga perlu siapkan anotasi "Interception" seperti berikut sebagai Pointcut Annotation.
Interception
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Interception{
String printValue();
}
Setup Aspect
Sekarang kita akan coba bikin AOP menggunakan Pointcut annotation "Interception". Secara umum AOP memiliki 5 jenis Advice, yaitu Before, After, After Throwing, After Returning, dan Around. Masing-masing Advice tersebut di-konfigurasi pada class Aspect yang bertugas melakukan konfigurasi penyisipan behavior. Contohnya seperti berikut:
InterceptionConfig
@Aspect
public class InterceptionConfig{
}
Di sini kita perlu tambahkan anotasi @Aspect
untuk menandakan bahwa disinilah aspect tersebut dikonfigurasi. Oh ya, kalau menggunakan Spring AOP, begini saja tidak cukup. Kita juga perlu menjadikan objek tersebut Spring Bean seperti penjelasan sebelumnya. Kita bisa menambahkan anotasi @Configuration
pada class InterceptionConfig. Sekarang kita tinggal bikin Pointcut dan Advice yang ingin kita sisipkan ke dalam method Calculator.
@Before
Ini adalah Advice yang diimplementasi pada @BeforeEach
di JUnit. Sesuai namanya, ini artinya kita akan menyisipkan behavior tertentu sebelum method tersebut dieksekusi. Sekarang kita akan bikin method add
dan subtract
pada Calculator untuk melakukan print value “penambahan” dan “pengurangan” sebelum penambahan dan pengurangan itu dilakukan.
Calculator
@Interception(printValue = "penambahan")
public int add(int x, int y){
int result = x + y;
System.out.println("x + y = " + result);
return result;
}
@Interception(printValue = "pengurangan")
public int subtract(int x, int y){
int result = x - y;
System.out.println("x - y = " + result);
return result;
}
public int multiply(int x, int y){
int result = x * y;
System.out.println("x * y = " + result);
return result;
}
InterceptionConfig
@Before("@annotation(interception) && execution(* *(..))")
public void before(Interception interception){
System.out.println(interception.printValue());
}
Di sini kita tambahkan annotasi @Before
dengan Pointcut @annotation(interception)
sesuai nama annotasi pada paramater. Itu artinya setiap method yang memiliki annotasi @Interception
akan disisipkan println
sebelum dieksekusi. Juga sebaiknya tambahkan Pointcut execution(* *(..))
secara spesifik agar Pointcut yang dieksekusi adalah “execution”, karena selain “execution” juga ada Pointcut “call” untuk menghindari eksekusi dua kali. Tapi untuk Spring AOP ini tidak perlu karena Spring AOP hanya support Pointcut “execution”.
Contoh code
Calculator calculator = new Calculator();
calculator.add(2, 1);
calculator.subtract(2, 1);
calculator.multiply(2, 1);
Sekarang setiap kita eksekusi add
dan subtract
method maka akan muncul print “penambahan” dan “pengurangan” setiap sebelum eksekusi.
@After
Ini Advice yang diimplementasi pada @AfterEach
di JUnit. Ini kebalikan dari @Before
, yaitu menyisipkan behavior tertentu setelah method tersebut di-eksekusi. Baik ketika method tersebut kena exception maupun tidak. Contohnya seperti berikut, kita ingin print “executed” setelah eksekusi.
InterceptionConfig
@After("@annotation(interception) && execution(* *(..))")
public void after(Interception interception){
System.out.println("executed");
}
Sekarang setelah kita tambahkan code di atas, akan muncul “executed” setiap selesai eksekusi.
@AfterThrowing
Ini Advice yang diimplementasi pada @ControllerAdvice
di Spring Web. Bedanya dengan @After
adalah ini hanya akan dieksekusi ketika terjadi error. Misalnya saat melakukan pembagian dengan 0, maka kita ingin print “ada error”.
Calculator
@Interception(printValue = "pembagian")
public int divide(int x, int y){
int result = x / y;
System.out.println("x / y = " + result);
return result;
}
InterceptionConfig
@AfterThrowing("@annotation(interception) && execution(* *(..))")
public void afterThrowing(Interception interception){
System.out.println("ada error");
}
Contoh code
Calculator calculator = new Calculator();
calculator.divide(2, 0);
Sekarang setiap terjadi exception, maka akan muncul “ada error” pada console.
@AfterReturning
Ini adalah kebalikan dari @AfterThrowing
, yaitu hanya akan dieksekusi ketika method tersebut tidak ada error. Misalnya kita ingin print “tak ada error” setiap eksekusi tanpa ada exception.
InterceptionConfig
@AfterReturning("@annotation(interception) && execution(* *(..))")
public void afterReturning(Interception interception){
System.out.println("tak ada error");
System.out.println();
}
Contoh code
Calculator calculator = new Calculator();
calculator.divide(2, 2);
Sekarang jika tidak terjadi exception, maka akan muncul “tak ada error” pada console.
@Around
Ini diimplementasi oleh @Transactional
pada Spring Data. Ini adalah gabungan @After
dan @Before
di atas. Di sini kita perlu tambahkan parameter “ProceedingJoinPoint” untuk eksekusi method aslinya. Jadi method aslinya akan dieksekusi di tengah-tengah sisipan. Misalnya kita ingin print value “kalkulasinya dimulai” sebelum eksekusi dan print “kalkulasinya selesai” setelah eksekusi pada method multiply
. Biar ga conflict dengan anotasi sebelumnya, kita akan tambahkan anotasi baru.
InterceptionAround
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface InterceptionAround{
}
Calculator
@InterceptionAround
public int multiply(int x, int y){
int result = x * y;
System.out.println("x * y = " + result);
return result;
}
InterceptionConfig
@Around("@annotation(interceptionAround) && execution(* *(..))")
public Object around(ProceedingJoinPoint proceedingJoinPoint, InterceptionAround interceptionAround) throws Throwable{
System.out.println("kalkulasinya dimulai");
Object proceed = proceedingJoinPoint.proceed();
System.out.println("kalkulasinya selesai");
System.out.println();
return proceed;
}
Contoh code
Calculator calculator = new Calculator();
calculator.multiply(2, 2);
Kita perlu eksekusi proceedingJoinPoint.proceed()
untuk mengeksekusi method aslinya dan return hasilnya. Sekarang akan muncul value “kalkulasinya dimulai” sebelum eksekusi dan print “kalkulasinya selesai” setelah eksekusi method multiply
.
Verdict
Kita telah menerapkan AOP menggunakan AspectJ dan Spring AOP menggunakan Pointcut annotation. Selain menggunakan annotation, kita juga bisa menggunakan Pointcut parameter argument tertentu atau menggunakan pola tertentu. Tapi cara yang paling umum, mudah di-maintain, dan gampang dipelajari adalah menggunakan annotation. Pendekatan AOP sering diimplementasi oleh framework untuk mempermudah pengembangan seperti Spring dan JUnit. Menggunakan AspectJ lebih cepat karena bekerja secara native, tapi setup-nya lebih ribet dan sering ga compatible dengan library annotation processor lain. Menggunakan Spring AOP lebih simple dan compatible dengan library annotation processor lain, tapi lebih lambat karena berbasis Proxy yang dibuat secara runtime. Makanya sebaiknya penggunaan AOP ini dihindari untuk kasus yang sangat sederhana karena terlalu overkill dan mubazir. AOP cocoknya digunakan pada kasus dengan pola tertentu yang berulang dan cukup kompleks agar mudah dimaintain. Code di atas gw share di github gw menggunakan AspectJ dan Spring AOP.