gpt4 book ai didi

java - 使用通用接口(interface)时不调用 EJB 拦截器

转载 作者:搜寻专家 更新时间:2023-11-01 03:13:04 25 4
gpt4 key购买 nike

给定以下代码

public interface Foo<T> {
T get();
}

@Remote
public interface Bar extends Foo<String> {
}

@Stateless
public class BarImpl implements Bar {
@Interceptors(ExceptionInterceptor.class)
public String get() {
throw new RuntimeException("not implemented");
}
}

public class ExceptionInterceptor {
@AroundInvoke
public Object convertExceptionForExternalSystem(InvocationContext ctx) throws RuntimeException, Error {
try
{
return ctx.proceed();
}
catch (Throwable e)
{
if (e instanceof Error)
throw new Error("Changed");
throw new RuntimeException("Changed");
}
}
}

当我们调用远程方法时,

Bar bar = context.lookup(Bar.class.getName());
bar.get();

Foo foo = context.lookup(Bar.class.getName());
foo.get();

未调用拦截器(使用 Glassfish 3.0.1)。

问题好像是接口(interface)的编译类文件是

javap Foo
Compiled from "Foo.java"
public interface Foo{
public abstract java.lang.Object get();
}

对于 BarImpl 来说是

javap BarImpl
Compiled from "BarImpl.java"
public class BarImpl extends java.lang.Object implements Bar{
public BarImpl();
public java.lang.String get();
public java.lang.Object get();
}

所以,当我们调用

Bar bar = ...;
bar.get();

内部方法

public java.lang.Object get(); 

被调用,它将委托(delegate)给

public java.lang.String get();

拦截器似乎只有在直接调用后者时才会被调用。当我将界面 Bar 更改为

@Remote
public interface Bar extends Foo<String> {
@Override
String get();
}

拦截器在第一次调用(bar.get())中被调用,但在第二次调用(foo.get())中没有被调用。在类级别定义拦截器可能会解决问题,但在我们的案例中不是一个选项。

我们做错了什么,或者这是 java-ee-6 的普遍问题,还是 glassfish 中的错误?有解决方法吗?还是我们应该完全放弃在我们的服务中使用泛型?

最佳答案

由于我们处理的是 Java,类型删除在运行时接管,您的业务接口(interface)将永远只包含:

public Object get();

您正在将接口(interface)转换为 Bar 并且没问题(没有运行时类转换异常)但是被调用的业务方法是业务接口(interface)中存在的方法(因为这是客户端的唯一方法知道):Object get() 版本。

此外,由于按照设计,只有业务方法调用会被拦截,Object get() 委托(delegate) String get() 不会被拦截(它只是一个方法调用另一个方法)。这意味着,在您的初始场景中,您试图拦截不属于业务接口(interface)的方法,因此永远不会被拦截。

由于 EJB3 规范对泛型确实不清楚(几乎什么都没说),实现实际上将这部分委托(delegate)给了 JVM。在这种情况下无法获得真正的通用支持,我会说您被类拦截器或更改业务接口(interface)所困。

关于java - 使用通用接口(interface)时不调用 EJB 拦截器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5337879/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com