繁体   English   中英

Java泛型在Eclipse中编译,而不是从命令行编译

[英]java generics compiles in eclipse and not from commandline

我知道类似的问题已经在下面的几个线程中进行了讨论,但是我仍然无法找到解决问题的方法。

其他线程: 编译器的行为与泛型方法的null参数不同

Java泛型代码在Eclipse中编译,但不在命令行中编译

因此,我正在使用事件和事件处理程序接口的泛型编写事件机制。 我有以下代码用于注册在命令行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中修复

http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=6369605

我认为我们没有足够的详细信息来回答您的第一个问题。 但是,对于第二个问题,您可以使用首选项-> 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.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM