如何使用标准java功能拦截方法调用(没有AspectJ等)?

时间:2009-02-23 08:43:33

标签: java reflection methods

我想拦截对某些类MyClass的所有方法调用,以便能够对某些setter-invocations做出反应。

我尝试使用动态代理,但据我所知,这仅适用于实现某些接口的类。但MyClass没有这样的界面。

除了实现一个包装类之外,还有其他方法可以将所有调用委托给一个成员,该成员是MyClass的一个实例,或者是使用AOP的旁边?

7 个答案:

答案 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)

  1. 为什么你的类不能实现接口?您可以从中提取一些包含您要拦截的所有方法的接口,并轻松使用动态代理机制。使用接口而不是类进行编码也是一种很好的编程习惯。

  2. 您可以将Spring框架与Spring AOP功能(内部使用动态代理)一起使用。您只需要在配置文件中将您的类定义为Spring bean,并且您的类的客户端必须从Spring应用程序上下文中获取其实例或自动作为依赖项(例如,通过定义setMyClass(MyClass mc)方法) )。从那里,您可以轻松地定义一个截取对此类的所有方法调用的方面。