[英]java generics compiles in eclipse and not from commandline
我知道类似的问题已经在下面的几个线程中进行了讨论,但是我仍然无法找到解决问题的方法。
其他线程: 编译器的行为与泛型方法的null参数不同
因此,我正在使用事件和事件处理程序接口的泛型编写事件机制。 我有以下代码用于注册在命令行javac中生成错误的事件,但与eclipse indigo无关。 在下面的代码中, EventRegistry
仅捕获字符串格式的事件和事件处理程序的类名称。
我创建了以下接口
public interface Event{}
public interface EventHandler<T extends Event> {}
我的EventManager类中的方法
public <T extends Event, P extends EventHandler<T>> void registerManagedHandler( EventRegistry er, ClassLoader cl ) throws ClassNotFoundException
{
...
Class<T> eventClass = (Class<T>) cl.loadClass(er.getEventClass());
Class<P> eventHandlerClass = (Class<P>) cl.loadClass(er.getEventHandlerClass());
...
register(eventHandlerClass, eventClass);
}
private <T extends Event, P extends EventHandler<T>> void register( Class<P> handler, Class<T> eventClass )
{
...
}
...在其他班级。 我有个说法。
EventManager.getInstance().registerManagedHandler(er,al);
Eclipse可以正确编译代码,但是在命令行上从javac编译时会生成错误
incompatible types; inferred type argument(s) com.mycompany.events.Event,java.lang.Object do not conform to bounds of type variable(s) T,P
[javac] found : <T,P>void
[javac] required: void
[javac] EventManager.getInstance().registerManagedHandler(er,al);
有类似的方法可以unregistering/enabling/disabling
产生相同错误的事件。 从上面的代码中,我尝试从registerManagedHandler删除类型约束( <T extends Event, P extends EventHandler<T>> )
,但是然后register(...)方法调用生成错误
我修改过的registeredManagedHandler()。
public void registerManagedHandler( EventRegistry er, ClassLoader cl ) throws ClassNotFoundException
{
...
Class<? extends Event> eventClass = (Class<? extends Event>) cl.loadClass(er.getEventClass());
Class<? extends EventHandler<? extends Event>> eventHandlerClass = (Class<? extends EventHandler<? extends Event>>) cl.loadClass(er.getEventHandlerClass());
...
register(eventHandlerClass, eventClass);
}
新生成的编译时eclipse中的错误。
Bound mismatch: The generic method register(Class<P>, Class<T>) of type EventManager is not applicable for the arguments
(Class<capture#20-of ? extends EventHandler<? extends Event>>, Class<capture#22-of ? extends Event>). The inferred type
capture#20-of ? extends EventHandler<? extends Event> is not a valid substitute for the bounded parameter <P extends
EventHandler<T>>
我不打算从register(...)方法中删除类型检查。
如果需要了解更多代码细节,请告知我。
请告诉我正确的方式来处理此类问题或解决方法。 我对泛型不是很陌生,但是在实现此功能之前,我已阅读了一些基本指南。
同样,我也找不到任何方法来强制Eclipse使用我的Ubuntu系统上安装的sun-javac6而不是其自己的编译器。 虽然我知道如何在Eclipse中更改JRE(“项目”->“属性”->“ Java构建路径”->“库”)
提前致谢。
更新:谢谢大家的答复。 告诉我是否可以提供更多信息。 我的Eclipse版本是Indigo(3.7)
这是sscce。 如果您在eclipse中运行程序,则可以正常工作(编译和执行)。 但是,当您使用命令行运行它:即javac GenericTester.java时,就会出现以下错误。 我已经使用sscce.org的编译器工具确认了这一点。
interface Event {
}
interface EventHandler<T extends Event> {
}
class EventRegistry {
public String eventClass;
public String eventHandlerClass;
}
class EventManager {
public <T extends Event, P extends EventHandler<T>> void registerEvent(
Class<T> eventClass, Class<P> eventHandlerClass) {
}
public <T extends Event, P extends EventHandler<T>> void registerEvent(
EventRegistry er) throws ClassNotFoundException {
Class<T> eventClass = (Class<T>) this.getClass()
.getClassLoader().loadClass(er.eventClass);
Class<P> eventHandlerClass = (Class<P>) this
.getClass().getClassLoader()
.loadClass(er.eventHandlerClass);
registerEvent(eventClass, eventHandlerClass);
}
}
class MyEvent implements Event {
}
class MyEventHandler implements EventHandler<MyEvent> {
}
public class GenericTester {
public static void main(String[] args) {
EventRegistry er = new EventRegistry();
er.eventClass = MyEvent.class.getName();
er.eventHandlerClass = MyEventHandler.class.getName();
try {
new EventManager().registerEvent(er);
System.out.println("It worked.");
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
来自命令行的错误,包括我运行的命令:
nitiraj@pandora:~/mywork/GenericTest/src$ javac GenericTester.java
GenericTester.java:40: incompatible types; inferred type argument(s) Event,java.lang.Object do not conform to bounds of type variable(s) T,P
found : <T,P>void
required: void
new EventManager().registerEvent(er);
^
Note: GenericTester.java uses unchecked or unsafe operations.
Note: Recompile with -Xlint:unchecked for details.
1 error
有关我已安装的Java的更多信息。
nitiraj@pandora:~/mywork/GenericTest/src$ java -version
java version "1.6.0_30"
Java(TM) SE Runtime Environment (build 1.6.0_30-b12)
Java HotSpot(TM) 64-Bit Server VM (build 20.5-b03, mixed mode)
nitiraj@pandora:~/mywork/GenericTest/src$ javac -version
javac 1.6.0_30
这是Java 5和Java 6中的错误,已在Java 7中修复
我认为我们没有足够的详细信息来回答您的第一个问题。 但是,对于第二个问题,您可以使用首选项-> Java->已安装的JRE配置Java运行时环境,并且可以使用首选项-> Java->编译器配置Java编译器符合性级别。
为什么JDK 1.6拒绝编译SSCCE的解释如下: 编译器的行为与通用方法的null参数不同
在Java 7中,对未出现在参数类型或返回值中的类型变量的推理算法进行了扩展,以考虑到类型参数的范围。 实际上,您的SSCCE可以使用JDK 1.7.0_02中的javac成功编译。
就是说,我认为您的registerEvent
滥用泛型:方法的类型参数由调用方指定(通常由类型推断算法指定,但只能明确指定),但是您的代码仅在类型正确时才使用调用方与EventRegistry的匹配,但编译器不会检查。 考虑:
er.eventHandlerClass = "java.lang.String";
new EventManager().<MyEvent, MyEventHandler>registerEvent(er);
可以编译并正常运行,但实际上触发事件时可能会导致ClassCastException。 这就是为什么我不建议将类作为字符串传递的原因。
如果您确实需要将它们作为String以便与ClassLoader结合在一起,则至少应检查这些类是否实现了正确的类型:
public void registerEvent(EventRegistry er) throws ClassNotFoundException {
Class<? extends Event> eventClass = this.getClass()
.getClassLoader().loadClass(er.eventClass)
.asSubclass(Event.class);
Class<? extends EventHandler> eventHandlerClass = this
.getClass().getClassLoader()
.loadClass(er.eventHandlerClass)
.asSubclass(EventHandler.class);
registerEvent(eventClass, eventHandlerClass);
}
通过该实现,尝试将String
订阅为EventHandler
将导致ClassCastException
。
该代码仍然不是完美的,因为编译器警告我们注意以下事实:我没有检查EventHandler是否具有正确的type参数,因此调用者仍然可以这样做:
er.eventClass = Event.class.getName();
er.eventHandlerClass = MyEventHandler.class.getName();
new EventManager().registerEvent(er);
即使MyEventHandler
不能处理所有Event
。 但是,由于类型擦除,通常无法验证EventHandler
的事件类型。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.