Java Generic bináris keresési fa típusú kérdés

szavazat
1

Dolgozom ezen házi Ez a fajta zavaró számomra ...

Én látva a következő BinarySearchTree osztály

import java.util.NoSuchElementException;

/**
 *
 * @param <T> The type of data stored in the nodes of the tree, must implement  Comparable<T> with the compareTo method.
 */
public class BinarySearchTree<T extends Comparable<T>> {


    BinaryTree<T> tree;

    int size;
    public BinarySearchTree() {
        tree = new BinaryTree<T>();
        size = 0;
    }

    public boolean isEmpty() {
        return tree.isEmpty();
    }

    protected BinaryTree<T> recursiveSearch(BinaryTree<T> root, T key) {
        if (root == null) {
            return null;
        }
        int c = key.compareTo(root.data);
        if (c == 0) {
            return root;
        }
        if (c < 0) {
            return recursiveSearch(root.left, key);
        } else {
            return recursiveSearch(root.right, key);
        }
    }

    public T search(T key) {
        if (tree.isEmpty()) { 
            return null;
        }
        return recursiveSearch(tree, key).data;
    }

    public void insert(T item) {

        if (tree.isEmpty()) { // insert here
            tree.makeRoot(item);
            size++;
            return;
        }

        // do an iterative descent
        BinaryTree<T> root = tree;
        boolean done=false;
        BinaryTree<T> newNode = null;
        while (!done) {
            int c = item.compareTo(root.data);
            if (c == 0) { // duplicate found, cannot be inserted
                throw new OrderViolationException();
            }
            if (c < 0) { // insert in left subtree
                if (root.left == null) { // insert here as left child
                    newNode = new BinaryTree<T>();
                    root.left = newNode;
                    done=true;
                } else { // go further down left subtree
                    root = root.left;
                }
            } else { // insert in right subtree
                if (root.right == null) { // insert here as right child 
                    newNode = new BinaryTree<T>();
                    root.right = newNode;
                    done=true;
                } else { // go further down right subtree
                    root = root.right;
                }
            }
        }
        // set fields of new node
        newNode.data = item;
        newNode.parent = root;
        size++;
    }

    /**
     * @param deleteNode Node whose parent will receive new node as right or left child,
     *                  depending on whether this node is its parent's right or left child. 
     * @param attach The node to be attached to parent of deleteNode.
     */
    protected void deleteHere(BinaryTree<T> deleteNode, BinaryTree<T> attach) {

        // deleteNode has only one subtree, attach
        BinaryTree<T> parent = deleteNode.parent;
        deleteNode.clear();  // clear the fields
        if (parent == null) {
            return;
        }
        if (deleteNode == parent.left) {
            // left child of parent, attach as left subtree
            parent.detachLeft();
            parent.attachLeft(attach);
            return;
        }
        // attach as right subtree
        parent.detachRight();
        parent.attachRight(attach);
    }


    protected BinaryTree<T> findPredecessor(BinaryTree<T> node) {
        if (node.left == null) {
            return null;
        }
        BinaryTree<T> pred = node.left; // turn left once
        while (pred.right != null) { // keep turning right
            pred = pred.right;
        }
        return pred;
    }


    public T delete(T key) {
        if (tree.isEmpty()) { // can't delete from an empty tree
            throw new NoSuchElementException();
        }

        // find node containing key 
        BinaryTree<T> deleteNode = recursiveSearch(tree, key);
        if (deleteNode == null) { // data not found, can't delete
            throw new NoSuchElementException();
        }

        BinaryTree<T> hold;

        // case c: deleteNode has exactly two subtrees
        if (deleteNode.right != null && deleteNode.left != null) {
            hold = findPredecessor(deleteNode);
            deleteNode.data = hold.data;
            deleteNode = hold; // fall through to case a or b
        }

        // case a: deleteNode is a leaf
        if (deleteNode.left == null && deleteNode.right == null) {
            deleteHere(deleteNode, null);
            size--;
            return deleteNode.data;
        }       

        // case b: deleteNode has exactly one subtree
        if (deleteNode.right != null) {
            hold = deleteNode.right;
            deleteNode.right = null;
        } else {
            hold = deleteNode.left;
            deleteNode.left = null;
        }

        deleteHere(deleteNode,hold);
        if (tree == deleteNode) { // root deleted
            tree = hold;
        }
        size--;
        return deleteNode.data;
    }


    public T minKey() {
        if (tree.data == null) { // tree empty, can't find min value
            throw new NoSuchElementException();
        }

        BinaryTree<T> root = tree;
        T min=root.data;
        root = root.left;  // turn left once
        while (root != null) {  // keep going left to leftmost node
            min = root.data;
            root = root.left;
        }
        return min;
    }


    public T maxKey() {
        if (tree.getData() == null) { // tree empty, can't find max value
            throw new NoSuchElementException();
        }

        BinaryTree<T> root=tree;
        T max=root.data;
        root = root.right;  // turn right once
        while (root != null) { // keep going to rightmost node
            max = root.data;
            root = root.right;
        }
        return max;
    }


    public int size() {
        return size;
    }


    protected void recursivePreOrder(BinaryTree<T> root, Visitor<T> visitor) {
        if (root != null) {
            visitor.visit(root);
            recursivePreOrder(root.left, visitor);
            recursivePreOrder(root.right, visitor);
        }
    }


    public void preOrder(Visitor<T> visitor) {
        if (tree.isEmpty()) {
            return;
        }
        recursivePreOrder(tree, visitor);
    }


    protected void recursiveInOrder(BinaryTree<T> root, Visitor<T> visitor) {
        if (root != null) {
            recursiveInOrder(root.left, visitor);
            visitor.visit(root);
            recursiveInOrder(root.right, visitor);
        }
    }


    public void inOrder(Visitor<T> visitor) {
        if (tree.isEmpty()) {   
            return;
        }
        recursiveInOrder(tree, visitor);
    }


    protected void recursivePostOrder(BinaryTree<T> root, Visitor<T> visitor) {
        if (root != null) {
            recursivePostOrder(root.left, visitor);
            recursivePostOrder(root.right, visitor);
            visitor.visit(root);
        }
    }

    public void postOrder(Visitor<T> visitor) {
        if (tree.isEmpty()) {
            return;
        }
        recursivePostOrder(tree, visitor);
    }
}

================================================== ==============================

Most van egy másik osztály Student .... Szeretnék létrehozni egy bináris keresési fa Student tárgyak ..

BinarySearchTree<Student> tree = new BinarySearchTree<Student>();

Azonban, ha ilyet kapok a következő hiba:

Bound mismatch: A típus Student nem helyettesíthetik a korlátos paraméter> típusú BinarySearchTree

Bármilyen ötletet, hogy mi történik itt ... nem tudok rájönni.

A kérdést 02/05/2009 06:31
a forrás felhasználó
Más nyelveken...                            


3 válasz

szavazat
0

Vajon az osztály Student végre Összehasonlítható?

Válaszolt 02/05/2009 06:41
a forrás felhasználó

szavazat
0

de nem vagyok egészen biztos abban, hogyan hajtsák végre a compareTo módszer.

Alapvetően ez olyasmi, mint a következő. Hogy a rendelési működik meg kell dönteni.

class Student implements Comparable<Student> {

    //...

    int compareTo(Student other) {
        // return some negative number if this object is less than other
        // return 0 if this object is equal to other
        // return some positive number if this object is greater than other
    }
}
Válaszolt 02/05/2009 06:56
a forrás felhasználó

szavazat
6

 public class BinarySearchTree<T extends Comparable<T>> 

A formális generikumok érv, az Ön esetében a T felsorolja, mi szükséges egy osztály egy érvényes T. Ön esetében, akkor már azt mondta: „ahhoz, hogy érvényes T, egy osztály végre kell hajtania összehasonlítható” (a kulcsszó a „terjed ”, de a gyakorlatban azt jelenti, hogy»kiterjeszti vagy eszközök«.)

Az Ön példányosítását, T Student. Ha helyettesíteni Student T:

public class BinarySearchTree<Student extends Comparable<Student>>

hogy igaz állítás? Vajon tényleg Student végre Összehasonlítható?

Ha igen, akkor Student illik a követelménynek, hogy a T, és így használhatja Student mint az aktuális paraméter a formális paraméter T.

Ha nem, akkor kap a fordító panaszát láttál.

Tulajdonképpen fedezésére bonyolultabb helyzetekben, amikor egy alosztály megvalósításának hasonló történik szuper osztály, annál általánosabb formában lenne:

   public class BinarySearchTree<T extends Comparable<? super T > > 

Tehát meg kell tenni Student végre összehasonlítható <Student>.

Figyeljük meg, hogy én nem mondom, hogy a fordító keres egy Student.compareTo. Ez még csak nem is olyan messze. Úgy néz, hogy ha T (az Ön esetében, diák) nyilvánították végrehajtási összehasonlítható <T> (az Ön esetében, összehasonlítható <Student>).

Most hozzátéve implements Comparable< Student >, hogy Hallgató is , hogy a fordító arról, hogy van egy public int compareTomódszer Student. De anélkül, hogy a „végrehajtja összehasonlítható”, még akkor is, ha a fordító tudja, hogy van egy módszer Student.compareTo, hogy nem tudom, hogy compareToaz Comparable.compareTo.

(Más szavakkal, keresünk bejelentett végrehajtására, nem csak, hogy előfordul, hogy a módszer a megfelelő neve és aláírása.)

Válaszolt 02/05/2009 06:57
a forrás felhasználó

Cookies help us deliver our services. By using our services, you agree to our use of cookies. Learn more