4

在 Java 中使用泛型时,我遇到了一些不便。请考虑以下代码:

/**
 * MyElement class is simply a wrapper for a generic object.
 */

public static class MyElement<T> {
    public final T OBJ;

    public MyElement(T obj) {
        this.OBJ = obj;
    }
}

/**
 * MyElementList contains an array list of MyElements of the given type, T.
 * This represents a class that uses a list of MyElements of a certain type,
 * and this list can be accessed in an unmodifiable format.
 */

public static class MyElementList<T> {

    //Properties
    private List<MyElement<T>> elementList = new ArrayList();

    //CTOR
    public MyElementList(List<MyElement<T>> initElements) {
        elementList.addAll(initElements);
    }

    //Getter
    public List<MyElement<T>> getElements() {
        return Collections.unmodifiableList(elementList);
    }

}

public static void main(String[] args) {
    //New list of elements
    //Notice that I did not explicitly specify the type for 'MyElement'
    List<MyElement> theElements = new ArrayList(Arrays.asList(
                new MyElement[] {
                    new MyElement("E 1"),
                    new MyElement("E 2"),
                    new MyElement("E 3")
                }
            ));

    //Also notice I did not explicitly specify the type for 'MyElementList'
    MyElementList theList = new MyElementList(theElements);

    //The following does not work.
    //It seems to not work because theList.getElements() returns a 'List'
    //not necessarily a 'List<MyElement>' which is what I would expect it to
    //return...
    //Why???

    for(MyElement e : theList.getElements()) {
        System.out.println(e.OBJ.toString());
    }

    //Currently my work around is to do the following, but I do not like 
    //having to introduce another variable, and I would rather just do the 
    //one above

    List<MyElement> listOfElements = theList.getElements();
    for(MyElement e : listOfElements) {
        System.out.println(e.OBJ.toString());
    }

    //How come the first 'for-each' loop method does not work?
    //Is there anyway I could get it to work?
    //THANK YOU!
}

在 main 方法中,如果我没有为“MyElementList”指定类型参数,那么“getElements()”方法只返回一个“List”,而不是一个“List<MyElement>”。这很不方便,因为如果我想遍历每个“MyElement”,我需要引入另一个变量作为临时列表,如代码所示。

  • 为什么“getElements()”方法不返回“List<MyElement>”?
  • 在不对“MyElementList”进行重大更改的情况下,我能做些什么来解决这个问题吗?
  • 这是一个糟糕的设计实践吗?

我使用的 IDE 是 Netbeans 7.2

提前致谢!

编辑

谢谢大家的快速回复。我对这里的社区印象深刻。我得出以下结论:

  • 如果没有指定泛型提示,Java 会忽略一个类的所有其他相关的泛型提示——这有点蹩脚,但我可以忍受。
  • 使用泛型时,最佳实践是在创建类的实例时实际指定泛型类型。这似乎是最面向对象的解决方案。

再次感谢!

4

4 回答 4

2

如果你改变MyElementList看起来像

public static class MyElementList<T extends MyElement> {

  //Properties
  private List<T> elementList = new ArrayList<T>();

  //CTOR
  public MyElementList(List<T> initElements) {
      elementList.addAll(initElements);
  }

  //Getter
  public List<T> getElements() {
      return Collections.unmodifiableList(elementList);
  }
}

它应该工作。

编辑泛型可以看作是 Java 中的编译时提示,因为 Java 擦除会将泛型转换为 Object。如上所述更新您的类将告诉编译器仅扩展MyElement适合列表并且for(MyElement e : theList.getElements())可以工作的元素。

编辑 2正如其他人所指出的(对不起,我乍一看没有看到)也将原始声明更改为:

MyElementList<MyElement> theList = new MyElementList<MyElement>(theElements);
于 2013-01-29T15:47:13.433 回答
1

而不是使用

for(MyElement e : theList.getElements()) {
    System.out.println(e.OBJ.toString());
}

你可以使用

for (Iterator<MyElement> it = theList.getElements().iterator(); it.hasNext();) {
    MyElement e = it.next();
    System.out.println(e.next().OBJ.toString());
}

这使您的编译器兼容。

但我更愿意指定你的类在实例化/访问它们时需要的类型(我猜你的编译器也是如此;))。

于 2013-01-29T15:52:08.313 回答
1

Te first 不起作用,因为getElements返回List<?>原始类型的 a

第二个有效,因为您将其分配给 a List<MyElement>,而忽略了警告。忽略是可以的,因为你知道它包含什么,但编译器不知道。

于 2013-01-29T15:53:25.017 回答
0

为什么该getElements()方法不返回aList<MyElement>

因为MyElement是打字!

在不进行重大更改的情况下,MyElementList我可以做些什么来解决这个问题吗?

您可能可以使用通配符:

List<MyElement<?>> someList = getElements();
于 2013-01-29T16:00:37.043 回答