我遇到了以下问题。
我在包a
中有两个不同的包我希望在包b
中调用接口的实现方法,但实现类具有包可见性。
所以简化的代码如下所示:
package b;
public final class Factory {
public static B createB() {
return new ImplB();
}
public interface B {
void method();
}
static class ImplB implements B {
public void method() {
System.out.println("Called");
}
}
}
和祈求者:
package a;
import java.lang.reflect.Method;
import b.Factory;
import b.Factory.B;
public final class Invoker {
private static final Class<?>[] EMPTY_CLASS_ARRAY = new Class<?>[] {};
private static final Object[] EMPTY_OBJECT_ARRAY = new Object[] {};
public static void main(String... args) throws Exception {
final B b = Factory.createB();
b.method();
final Method method = b.getClass().getDeclaredMethod("method", EMPTY_CLASS_ARRAY);
method.invoke(b, EMPTY_OBJECT_ARRAY);
}
}
当我启动程序时,它按预期打印出Called
并抛出异常,因为包可见性禁止调用已发现的方法。
所以我的问题是解决这个问题的方法吗?我是否遗漏了Java文档中的内容,或者这根本不可能,尽管只是在没有反射的情况下调用已实现的方法。
答案 0 :(得分:2)
由于类具有包可见性,因此其方法可见性实际上是包级别而不是您认为的公共级别。
因此,您必须在方法调用之前调用method.setAccessible(true)
:
final Method method = b.getClass().getDeclaredMethod("method", EMPTY_CLASS_ARRAY);
method.setAccessible(true);
method.invoke(b, EMPTY_OBJECT_ARRAY);
答案 1 :(得分:1)
我发现了。我需要在已实现的接口上调用getDeclaredMethod
,而不是在类上。
示例:
final Method method = B.class.getDeclaredMethod("method", EMPTY_CLASS_ARRAY);
method.invoke(b, EMPTY_OBJECT_ARRAY);
答案 2 :(得分:0)
请参阅setAccessible的文档。 SecurityPolicy可能会阻止您访问它。
但目前尚不清楚你是否需要反思。多态的一个好处是客户端可以调用B.method()
而无法访问(甚至担心)实现接口的具体类。