Ответ 1
Как уже сказали другие, используйте DynamicProxy. Вот пример.
Этот класс использует DynamicProxy для перехвата вызовов методов, объявленных в интерфейсе "HammerListener". Он выполняет некоторые протоколирования, а затем делегирует "реальную" реализацию HammerListener (да, то же самое можно сделать с AOP).
См. метод newInstance для создания прокси-сервера (обратите внимание, что вам необходимо передать интерфейсы, которые должен реализовать прокси-сервер, - прокси-сервер может реализовать несколько интерфейсов).
Все вызовы методов на интерфейсах, которые реализует прокси-сервер, будут вызваны вызовом метода "invoke", который объявлен в интерфейсе "InvocationHandler". Все обработчики прокси должны реализовать этот интерфейс.
import java.lang.reflect.*;
/**
* Decorates a HammerListener instance, adding BEFORE/AFTER
* log messages around all methods exposed in the HammerListener interface.
*/
public class HammerListenerDecorator implements InvocationHandler {
private final HammerListener delegate;
static HammerListener newInstance(HammerListener delegate) {
ClassLoader cl = Thread.currentThread().getContextClassLoader();
return (HammerListener)Proxy.newProxyInstance(cl, new Class[]{HammerListener.class},
new HammerListenerDecorator(delegate));
}
private HammerListenerDecorator(HammerListener delegate) {
this.delegate = delegate;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
logger.info("BEFORE " + method.getName() + " {{{" + argsToString(args) + "}}}");
Object rtn = method.invoke(delegate, args);
logger.info("AFTER " + method.getName());
return rtn;
}
private String argsToString(Object[] args) {
StringBuilder sb = new StringBuilder();
for (Object o : args) {
sb.append(String.valueOf(o)).append(" ");
}
return sb.toString();
}
}