Как использовать AOP с AspectJ для регистрации? - PullRequest
27 голосов
/ 12 января 2012

Я хотел бы добавить «трассировочные» сообщения ко всем моим открытым методам следующим образом:

public void foo(s:String, n:int) { // log is a log4j logger or any other library
  log.trace(String.format("Enter foo with s: %s, n: %d", s, n))
  ...
  log.trace("Exit foo") 
}

Теперь я хотел бы добавить все эти log.trace в мои методы автоматически с помощью AOP (и байтового кода).измерительные приборы).Я думаю о AspectJ.Имеет ли это смысл?Вы знаете какой-нибудь open-source, который делает именно это?

Ответы [ 5 ]

27 голосов
/ 13 января 2012

Я создал простой аспект для захвата выполнения открытых методов.Ядром этого кода AspectJ является определение pointcut:

pointcut publicMethodExecuted(): execution(public * *(..));

Здесь мы собираем все открытые методы с любым типом возврата, для любого пакета и любого класса, с любым количеством параметров.

Выполнение совета может быть визуализировано в следующем фрагменте кода:

after(): publicMethodExecuted() {
    System.out.printf("Enters on method: %s. \n", thisJoinPoint.getSignature());

    Object[] arguments = thisJoinPoint.getArgs();
    for (int i =0; i < arguments.length; i++){
        Object argument = arguments[i];
        if (argument != null){
            System.out.printf("With argument of type %s and value %s. \n", argument.getClass().toString(), argument);
        }
    }

    System.out.printf("Exits method: %s. \n", thisJoinPoint.getSignature());
}

Этот совет использует thisJoinPoint для получения сигнатуры метода и аргументов.И это все.Вот код аспекта:

public aspect LogAspect {

pointcut publicMethodExecuted(): execution(public * *(..));

after(): publicMethodExecuted() {
    System.out.printf("Enters on method: %s. \n", thisJoinPoint.getSignature());

    Object[] arguments = thisJoinPoint.getArgs();
    for (int i =0; i < arguments.length; i++){
        Object argument = arguments[i];
        if (argument != null){
            System.out.printf("With argument of type %s and value %s. \n", argument.getClass().toString(), argument);
        }
    }
    System.out.printf("Exits method: %s. \n", thisJoinPoint.getSignature());
}

Для более сложных примеров я бы порекомендовал книгу AspectJ: в действии .

25 голосов
/ 03 февраля 2013

@Loggable аннотация и аспект AspectJ из jcabi-aspect - это готовый механизм для вас (я разработчик):

@Loggable(Loggable.DEBUG)
public String load(URL url) {
  return url.openConnection().getContent();
}

Для регистрации входа и выхода в соответствии с требованиями вопроса:

@Loggable(Loggable.DEBUG, prepend=true)
public String load(URL url) {
  return url.openConnection().getContent();
}

Все журналы идут на SLF4J.Проверьте этот пост для более подробной информации.

4 голосов
/ 12 января 2012

Вы можете использовать различные pointcut, чтобы сделать ваше требование. Эта документация поможет вам.

Прямое прямое решение

1 голос
/ 04 сентября 2013

Вы можете попробовать этот открытый исходный код http://code.google.com/p/perfspy/. PerfSpy - это инструмент для ведения журналов, мониторинга производительности и проверки кода. Он использует ApsectJ для обхода кода вашего приложения во время выполнения и регистрирует время выполнения каждого метода и его входные параметры и значения. Он имеет приложение пользовательского интерфейса, в котором вы можете просматривать вызовы методов и их входные и возвращаемые значения в виде деревьев. С его помощью вы можете обнаружить узкие места в производительности и понять сложный поток кода.

0 голосов
/ 21 декабря 2016

Вот моя простая реализация для регистрации входа, выхода и регистрации исключений из методов

Аннотация

package test;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ ElementType.METHOD, ElementType.TYPE })
public @interface Audit {

}

Перехватчик

import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.logging.Level;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.reflect.MethodSignature;


@Aspect
public class ExceptionInterceptor {

    private static final java.util.logging.Logger LOGGER = java.util.logging.Logger.getLogger(ExceptionInterceptor.class.getName());

    @Around("execution(* * (..))"
            + " && @annotation(test.Audit)"
    )
    public Object intercept(final ProceedingJoinPoint point) throws Throwable {
        final Method method
                = MethodSignature.class.cast(point.getSignature()).getMethod();
        String mName = method.getName();
        String cName = method.getDeclaringClass().getSimpleName();
        LOGGER.log(Level.INFO, "Entering {0}:{1}", new Object[]{cName, mName});
        Object out = null;
        try {
            out = point.proceed();
        } catch (Throwable t) {
            logExceptions(t, point);
        }
        LOGGER.log(Level.INFO, "Exiting {0}:{1}", new Object[]{cName, mName});
        return out;
    }

    private void logExceptions(Throwable t, final ProceedingJoinPoint point) {
        final Method method
                = MethodSignature.class.cast(point.getSignature()).getMethod();
        String mName = method.getName();
        String cName = method.getDeclaringClass().getSimpleName();
        Object[] params = point.getArgs();
        StringBuilder sb = new StringBuilder();
        sb.append("Exception caught for [");
        sb.append(cName);
        sb.append(".");
        sb.append(mName);
        for (int i = 0; i < params.length; i++) {
            Object param = params[i];

            sb.append("\n");
            sb.append("  [Arg=").append(i);
            if (param != null) {
                String type = param.getClass().getSimpleName();

                sb.append(", ").append(type);

                // Handle Object Array (Policy Override)
                if (param instanceof Object[]) {
                    sb.append("=").append(Arrays.toString((Object[]) param));
                } else {
                    sb.append("=").append(param.toString());
                }
            } else {
                sb.append(", null");
            }
            sb.append("]");
            sb.append("\n");
        }
        LOGGER.log(Level.SEVERE, sb.toString(), t);

    }
}

Как его использовать

@Audit  
public void testMethod(Int a,int b, String c){
}

Зависимости Maven Компиляция

    <dependency>
        <groupId>org.aspectj</groupId>
        <artifactId>aspectjrt</artifactId>
        <version>1.8.7</version>
    </dependency> 

Плетение

        <plugin>
            <groupId>com.jcabi</groupId>
            <artifactId>jcabi-maven-plugin</artifactId>
            <executions>
                <execution>
                    <phase>compile</phase>
                    <goals>
                        <goal>ajc</goal>
                    </goals>
                </execution>
            </executions>
        </plugin> 
...