0

このコードはアルゴリズムテキストからのものですが、ネストされたクラスとインターフェイスに関して少し問題があります。実際、私の混乱の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];
        }
    }

}
4

1 に答える 1

3

親クラスはIterableインターフェイスを実装しますが、ReverseArrayIteratorがIteratorを実装する場合、Iteratorインターフェイスはどこから直接取得されますか?

その質問はあまり意味がありません。IteratorインターフェースはJavaSEクラスライブラリから「取得」され(コードがインポートします)、ReverseArrayIteratorクラスによって実装されます。また、実行時に、メソッドを呼び出すと、クラスResizingArrayStack.iterator()のインスタンスが作成されます。ReverseArrayIterator

Intuitionは、Iteratorインスタンスメソッドから直接実装し、最終的にはIterableインターフェイスから実装していることを教えてくれます(拡張機能のようなものですか?)。

インターフェイスとの接続Iterableは、外部クラスがそれを実装することです。ただし、適切なタイプのインスタンスを作成するために呼び出すことができるメソッドがあることIterableを「意味」します。特別な「どのように機能を拡張するか」という魔法が発生することはありません。この例では、すべてインターフェイスを実装するクラスにすぎません。iterator()Iterator

これに関するもう1つの「異なる」点ReverseArrayIteratorは、ネストされたクラスであるため、親オブジェクトのプライベート状態にアクセスできることです。この場合aN

ここで起こっていることは、特定の全体的な効果を達成するために、いくつかの独立したもの(言語機能とデザインパターン)が組み合わされているということです。その全体的な効果は、「foreach」ループを使用してスタックの要素を反復できるようにすることです。

于 2012-06-11T03:53:06.750 に答える