虽然这段代码来自算法文本,但我对嵌套类和接口有点麻烦 - 实际上,90%的混淆源于此代码实现接口的方式。同样,这个问题与算法本身无关。
据我所知,此代码使用嵌套类,以便它可以访问ResizingArrayStack中的私有实例变量(此文本使用约定将所有实例变量声明为封装的私有)。
Iterable
界面是这样的:
public Iterator<Item> { Iterator<Item> iterator(); } // ignore the quotes
和Iterator
接口是这样的:
public interface Iterator<Item> { boolean hasNext(); Item next(); void remove(); }
我的问题是所有这些如何连接在下面显示的代码中。
父类实现了Iterable接口,但是当ReverseArrayIterator实现Iterator时,Iterator接口直接来自哪里?它来自Iterator实例方法还是来自Iterable接口? Intuition告诉我它是直接从Iterator实例方法实现的,最终是从Iterable接口实现的(有点像扩展工作吗?)。
很抱歉我缺乏OOP知识。本文只是简单地谈论它,我被告知我不必知道任何这些(我可能不会,只要我理解算法),但我必须理解它大声笑。我无法理解这一点。提前谢谢。
// from http://algs4.cs.princeton.edu/13stacks/ResizingArrayStack.java.html
import java.util.Iterator;
import java.util.NoSuchElementException;
public class ResizingArrayStack<Item> implements Iterable<Item> {
private Item[] a; // array of items
private int N; // number of elements on stack
// create an empty stack
public ResizingArrayStack() {
a = (Item[]) new Object[2];
}
public boolean isEmpty() { return N == 0; }
public int size() { return N; }
// resize the underlying array holding the elements
private void resize(int capacity) {
assert capacity >= N;
Item[] temp = (Item[]) new Object[capacity];
for (int i = 0; i < N; i++) {
temp[i] = a[i];
}
a = temp;
}
// push a new item onto the stack
public void push(Item item) {
if (N == a.length) resize(2*a.length); // double size of array if necessary
a[N++] = item; // add item
}
// delete and return the item most recently added
public Item pop() {
if (isEmpty()) { throw new RuntimeException("Stack underflow error"); }
Item item = a[N-1];
a[N-1] = null; // to avoid loitering
N--;
// shrink size of array if necessary
if (N > 0 && N == a.length/4) resize(a.length/2);
return item;
}
public Iterator<Item> iterator() { return new ReverseArrayIterator(); }
// an iterator, doesn't implement remove() since it's optional
private class ReverseArrayIterator implements Iterator<Item> {
private int i = N;
public boolean hasNext() { return i > 0; }
public void remove() { throw new UnsupportedOperationException(); }
public Item next() {
if (!hasNext()) throw new NoSuchElementException();
return a[--i];
}
}
}
答案 0 :(得分:3)
父类实现了Iterable接口,但是当ReverseArrayIterator实现Iterator时,Iterator接口直接来自哪里?
这个问题没有多大意义。 Iterator 接口“来自”Java SE类库('代码导入它),它由ReverseArrayIterator
类实现。在运行时,调用ResizingArrayStack.iterator()
方法时会调用ReverseArrayIterator
类的实例。
Intuition告诉我它是直接从Iterator实例方法实现的,最终是从Iterable接口实现的(有点像扩展工作吗?)。
与Iterable
接口的连接是外部类实现它。但是Iterable
只是“意味着”可以调用iterator()
方法来创建适当类型的Iterator
实例。没有特殊的“有点像扩展工作”的魔法发生。在这个例子中,它只是实现接口的类。
另一个“不同”的事情是ReverseArrayIterator
是一个嵌套类,因此可以访问父对象的私有状态。在这种情况下a
和N
。
这里发生的是,许多独立的事物(语言特征和设计模式)正在被组合以实现特定的整体效果。总体效果是可以使用“for each”循环迭代堆栈的元素。