再续java中的泛型

2025-10-26 15:04:10

1、先看实例:

再续java中的泛型

2、堆栈的实现在面试中非常流行。他们有一套相对较小的操作, 通常是push、pop, 也许还有peek。使用 java 集合 api 可以很容易地实现堆栈;不需要额外的库。花点时间, 确保你明白为什么在这里使用 linkedlist--arraylist 也能奏效吗?有什么区别?

3、对于给定的实现, 这是一个完全正常运行的堆栈, 尽管它尚未开发用于泛型。若要迁移此类以使用泛型, 可以使用编译器作为指南。首先, 必须声明 stack 类才能采用参数化类型:

public class GenericStack<E> {

 ...

4、e 是参数化类型变量。任何符号都可能被使用。这里的 e 代表 "元素", 反映了它在集合 api 中的使用。包含的列表现在可以使用此泛型类型。

private final List<E> values;

5、这将立即引发编译器错误。推送方法采用对象, 但现在需要获取 e 类型的对象:public void push(final E element) {     values.add(0, element); }

6、将值更改为 "List<E> 还会在 stack 构造函数中引发编译器警告。创建值 linkedlist 时, 这也应参数化:

public GenericStack() {

    values = new LinkedList<E>(); 

}

7、编译器不知道的一个更改是对 pop 方法的更改。pop 方法的最后一行, values.remove(0) 现在返回类型为 e. 当前, 此方法返回 object。编译器没有理由出错, 甚至没有理由抛出警告, 因为 object 是所有类的超级类型, 无论 e 是什么。可以将其更改为返回类型 e:

public E pop() {     

    if (values.size() == 0) {

         return null;

    }

    return values.remove(0);

 }

8、继承关系的类如何影响泛型呢?

9、给定以下类层次结构:

class A {} 

class B extends A {}

10、b 是 a 的子类型。但List<B> 不是List<A> 的子类型。java 的泛型系统被称为协方差, 无法对其进行建模。

11、在处理泛型类型时, 在某些时候, 您可能希望接受类的子类型。使用上一个问题中的 genericstack 类, 想象一个实用程序方法, 它从 a 列表中创建一个新的 genericstack:

public static GenericStack<A> pushAllA(final List<A> listOfA) {

     final GenericStack<A> stack = new GenericStack<>();

    for (A a : listOfA) {

         stack.push(a); 

    }

    return stack;

 }

12、这将完全按照 a 列表的预期方式进行编译和工作:

@Test

public void usePushAllA() {

    final ArrayList<A> list = new ArrayList<>();

    for (int i = 0; i < 10; i++) {

         list.add(new A()); 

    }

     final GenericStack<A> genericStack = pushAllA(list);

     System.out.println(genericStack.pop());

    }

}

13、但是, 当尝试添加 b 列表时, 它不会编译:

@Test

public void usePushAllAWithBs() {

    final ArrayList<B> listOfBs = new ArrayList<>();

    for(int i = 0; i < 10; i++) {

         listOfBs.add(new B());

    }

    final GenericStack<A> genericStack = pushAllA(listOfBs);

    System.out.println(genericStack.pop());

 }

14、虽然 b 是 a 的子类, 但List<B> 不是List<A> 的子类型。推送的方法签名需要更改以显式允许 a 和 a 的任何子类:

public static GenericStack<A> pushAllA(final List<? extends A> listOfA) {}

问号称为通配符, 它告诉编译器允许扩展 a 的类的任何实例。

声明:本网站引用、摘录或转载内容仅供网站访问者交流或参考,不代表本站立场,如存在版权或非法内容,请联系站长删除,联系邮箱:site.kefu@qq.com。
相关推荐
  • 阅读量:61
  • 阅读量:194
  • 阅读量:171
  • 阅读量:138
  • 阅读量:190
  • 猜你喜欢