A simple application to demonstrate how to implement Around Advice using @AspectJ Annotation-Driven AOP in Java.
Step 1:- Create a Interface Division.java
Step 2:- DivisionImpl.java Service Implementation Class
Step 3:- Around Advice Implementation class
Step 4 :- Spring Configuration file
Step 5:- Test class
Output of the program :
Step 1:- Create a Interface Division.java
package com.hubberspot.aspectj.annotation.aroundadvice;
// Its a simple interface for the Division service.
// It contains one single method called as divide().
// This method takes in two arguments both of the type
// int.
public interface Division {
public int divide(int a , int b);
}
Step 2:- DivisionImpl.java Service Implementation Class
package com.hubberspot.aspectj.annotation.aroundadvice;
// It is the implementation class for the
// Division service. It just calculates and returns
// division of two numbers passed to it as arguments.
public class DivisionImpl implements Division {
@Override
public int divide(int a, int b) {
System.out.println("Number are : " + a + ", " + b);
return a/b;
}
}
Step 3:- Around Advice Implementation class
package com.hubberspot.aspectj.annotation.aroundadvice;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
//@Aspect annotation treats this Java class as
//Aspect. Its not ordinary POJO class.
@Aspect
public class AroundDivideAdvice {
// Method invoke() is a around advice
// implemented by providing @Around annotation.
// This annotation takes in Pointcut expression, which
// indicates when this advice executes.
// This Pointcut expression tells that invoke() advice
// will execute around divide method of Division interface.
// The after advice takes in ProceedingJoinPoint which
// helps in invoking the divide method.
// Based on the user inputs the advice decides whether divide should
// be called or not.
// invoke() methods takes in ProceedingJoinPoint type as a argument
// It takes out arguments which user inputs from the ProceedingJoinPoint
// and than validate it accordingly.
// It also has a proceed() which transfers controls to method being advised.
@Around("execution(* com.hubberspot.aspectj.annotation.aroundadvice.Division.divide(..))")
public Object invoke(ProceedingJoinPoint joinPoint) throws Throwable {
Object arguments[] = joinPoint.getArgs();
int number = ((Integer)arguments[1]).intValue();
if (number == 0) {
throw new Exception("Cannot divide with 0.... ");
}
return joinPoint.proceed();
}
}
Step 4 :- Spring Configuration file
<?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:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.1.xsd"> <aop:aspectj-autoproxy></aop:aspectj-autoproxy> <!-- Implementation Class --> <bean id="divide" class="com.hubberspot.aspectj.annotation.aroundadvice.DivisionImpl" /> <bean id="aroundAdvice" class="com.hubberspot.aspectj.annotation.aroundadvice.AroundDivideAdvice" /> </beans> |
Step 5:- Test class
package com.hubberspot.aspectj.annotation.aroundadvice;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class DivisionTest {
public static void main(String[] args) {
// ApplicationContext is a Spring interface which
// provides with the configuration for an application.
// It provides us with all the methods that BeanFactory
// provides. It loads the file resources in a older
// and generic manner. It helps us to publish events to the
// listener registered to it. It also provides quick support
// for internationalization. It provides us with the object
// requested, it reads the configuration file and provides
// us with the necessary object required.
// We are using concrete implementation of AbstractApplicationContext
// here called as ClassPathXmlApplicationContext because this
// bean factory reads the xml file placed in the classpath of
// our application. We provide ClassPathXmlApplicationContext
// with a configuration file called as around_advice.xml placed
// at classpath of our application.
ApplicationContext context =
new ClassPathXmlApplicationContext("around_advice.xml");
// In order to get a object instantiated for a particular bean
// we call getBean() method of ClassPathXmlApplicationContext
// passing it the id for which the object is to be needed.
// Here getBean() returns an Object. We need to cast it back
// to the Divide object. Without implementing new keyword we
// have injected object of Divide just by reading an xml
// configuration file.
Division divide = (Division)context.getBean("divide");
int result = divide.divide(10 , 5);
System.out.println("Result = " + result);
result = divide.divide(10, 0);
System.out.println("Result = " + result);
}
}
Output of the program :
