拦截器绑定不起作用
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 文件。
我创建了一个自定义注释,如下所示
@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 文件。