我想拦截对某些类MyClass的所有方法调用,以便能够对某些setter-invocations做出反应。
我尝试使用动态代理,但据我所知,这仅适用于实现某些接口的类。但MyClass没有这样的界面。
除了实现一个包装类之外,还有其他方法可以将所有调用委托给一个成员,该成员是MyClass的一个实例,或者是使用AOP的旁边?
答案 0 :(得分:26)
如您所知,您不能使用JDK动态代理(无接口),但使用Spring和CGLIB(Spring包含JAR),您可以执行以下操作:
public class Foo
{
public void setBar()
{
throw new UnsupportedOperationException("should not go here");
}
public void redirected()
{
System.out.println("Yiha");
}
}
Foo foo = new Foo();
ProxyFactory pf = new ProxyFactory(foo);
pf.addAdvice(new MethodInterceptor()
{
public Object invoke(MethodInvocation mi) throws Throwable
{
if (mi.getMethod().getName().startsWith("set"))
{
Method redirect = mi.getThis().getClass().getMethod("redirected");
redirect.invoke(mi.getThis());
}
return null;
}
});
Foo proxy = (Foo) pf.getProxy();
proxy.setBar(); // prints "Yiha"
答案 1 :(得分:13)
如果你准备做一些非常难看的事情,请看看:
http://docs.oracle.com/javase/7/docs/technotes/guides/jpda/
基本上,调试器接口应该允许您像调试器一样附加,从而拦截调用。请记住,我认为这是一个真的坏主意,但你问是否有可能。
答案 2 :(得分:2)
Java没有用于方法拦截的任何实际语言功能(不确定任何静态语言)
我有点像Nick的使用调试器界面的想法,这只是意思。
我认为你需要的简短答案是:没有办法在没有用代理或包装器实际替换类的情况下拦截Java中的方法调用。
注意:AOP库只是自动发生这种情况。
答案 3 :(得分:2)
我刚刚为此目的开发了一个小框架。 你可以在http://code.google.com/p/java-interceptor/查看(使用svn签出)。
答案 4 :(得分:1)
一些Java大师可能对此赞不绝口,但我在完全避免原始类型和setter方面取得了一些成功。我的班级看起来像这样:
class Employee extends SmartPojo {
public SmartString name;
public SmartInt age;
}
你会注意到两件事:1。一切都是公开的。 2.没有构造函数。
魔术发生在SmartPojo中,它搜索实现“智能”界面并初始化它的任何字段。由于这不是原始的(并且没有最终类),我可以在一个地方中为模型中的任何位置的所有字段添加set()和get()方法。因此,不再需要setter / getter浪费,添加通知(也在一个地方)等非常简单。
是的,这不再是POJO了,而且在大多数情况下它都不是Bean,但我发现这些旧想法限制了我,而不是他们的帮助。 YMMV。
答案 5 :(得分:0)
AspectJ中没有太多魔力。您可以编写自己的代理。 http://java.sun.com/javase/6/docs/api/java/lang/instrument/package-summary.html似乎是一个很好的起点。
答案 6 :(得分:0)
为什么你的类不能实现接口?您可以从中提取一些包含您要拦截的所有方法的接口,并轻松使用动态代理机制。使用接口而不是类进行编码也是一种很好的编程习惯。
您可以将Spring框架与Spring AOP功能(内部使用动态代理)一起使用。您只需要在配置文件中将您的类定义为Spring bean,并且您的类的客户端必须从Spring应用程序上下文中获取其实例或自动作为依赖项(例如,通过定义setMyClass(MyClass mc)方法) )。从那里,您可以轻松地定义一个截取对此类的所有方法调用的方面。