拦截器绑定不起作用

InterceptorBinding is not working

我创建了一个自定义注释,如下所示

@InterceptorBinding
@Retention(RUNTIME)
@Target(TYPE, METHOD)
public @interface Traceable {}

我写了一个如下的拦截器

@Traceable
@Interceptor
public class EnterExitLogger {
    @AroundInvoke
    public Object aroundInvoke(InvocatiobContext c) {}
}

拦截器和注释位于名为 common-utils 的模块中。

我在 class 级别用 @Traceable 注释了我的目标 class,如下所示

@Traceable
public class CDIManagedBean {
}

我在 beans.xml 文件中声明了拦截器条目,如下所示

<interceptors>
    <class>my.package.EnterExitLogger</class>
</interceptors>

目标 class 在单独的模块中。 beans.xml 位于目标 class 模块的 META-INF 目录中。

目标 class 的方法是从 rest class 调用的。当我调用方法时,拦截器的 AroundInvoke 方法没有被调用。

我阅读了文档并了解到拦截器应该包含一个 public 无参数构造函数。我添加了它。但是还是没有调用拦截器。

我在阅读文档后在自定义注释上添加了@Inherited。但是还是没有调用拦截器。

从文档中我注意到拦截器实现了 Serializable 接口。虽然没有提到我也实现了 Serializable。还是不行。

然后我从拦截器、beans.xml 文件和目标 class 中删除了自定义注释。我还从拦截器中删除了 public 无参数构造函数并删除了 Serializable。

然后我用 @Interceptors(EnterExitLogger.class) 注释目标 class 并调用流程。我的拦截器被调用了。

谁能告诉我如何使用 InterceptorBinding?

P.S.

我正在 WAS 8.5 服务器中部署我的耳朵。

Java EE Tutorial 提供了一个很好的解释和一些关于拦截器的例子:

创建一个拦截器绑定注解,必须用@Inherited and @InterceptorBinding注解:

@Inherited
@InterceptorBinding
@Retention(RUNTIME)
@Target({METHOD, TYPE})
public @interface Logged { }

创建拦截器 class,它使用上面创建的拦截器绑定注释以及 @Interceptor 注释进行注释。

每个 @AroundInvoke method takes an InvocationContext argument, returns an Object, and throws an Exception. The @AroundInvoke method must call the InvocationContext#proceed() 方法,导致目标 class 方法被调用:

@Logged
@Interceptor
public class LoggedInterceptor implements Serializable {

    public LoggedInterceptor() {

    }

    @AroundInvoke
    public Object logMethodEntry(InvocationContext invocationContext) throws Exception {

        System.out.println("Entering method: "
                + invocationContext.getMethod().getName() + " in class "
                + invocationContext.getMethod().getDeclaringClass().getName());

        return invocationContext.proceed();
    }
}

定义拦截器和绑定类型后,您可以使用绑定类型注释 bean 和各个方法,以指定要在 bean 的所有方法或特定方法上调用拦截器。

例如,PaymentHandler bean被注释为@Logged,这意味着对其业务方法的任何调用都会导致拦截器的@AroundInvoke方法被调用:

@Logged
@SessionScoped
public class PaymentHandler implements Serializable {...}

但是,您可以通过仅注释所需的方法来仅拦截 bean 的一组方法:

@Logged
public String pay() {...}

@Logged
public void reset() {...}

为了在 CDI 应用程序中调用拦截器,必须在 beans.xml 文件中指定:

<interceptors>
    <class>your.package.LoggedInterceptor</class>
</interceptors>

如果一个应用程序使用多个拦截器,拦截器将按照 beans.xml 文件中指定的顺序调用。

在保留拦截器的模块中添加空 beans.xml 文件后,问题已得到解决。我认为只有当我想使用 cdi 注入时才需要 beans.xml 文件。