二进制搜索树与泛型实现的困难

时间:2015-12-06 22:16:22

标签: java

我在实现下面列出的BinarySearchTree时遇到问题。对于某些上下文,我正在创建一个基于接口的二进制搜索树,该接口需要泛型和类似的密钥。我认为代码中存在一个逻辑错误,它正在困扰我,它在BinarySearchTree中的insert方法中,但我不是百分百肯定。

下面是我的节点的类,它在我的BST中使用。

public class MyNodeClass<Key extends Comparable<Key>, Value>{
private Key key;
private Value value;
private MyNodeClass<Key,Value> left = null;
private MyNodeClass<Key,Value> right = null;

public MyNodeClass(Key key, Value val)
{
    this.key = key;
    this.value = val;
}

public void setKey(Key key){
    this.key = key;
}

public void setValue(Value value){
    this.value = value;
}

public Key getKey(){
    return this.key;
}

public Value getValue(){
    return this.value;
}

public void setLeft(MyNodeClass<Key, Value> l){
    this.left = l;
}

public void setRight(MyNodeClass<Key, Value> r){
    this.right = r;
}

public MyNodeClass<Key,Value> getLeft(){return this.left;}

public MyNodeClass<Key,Value> getRight(){return this.right;}

public int compareTo(Key that){
    return(this.getKey().compareTo(that));
}

}

public class MyBinarySearchTree<Key extends Comparable<Key>, Value> implements BinarySearchTree<Key,Value>  {

private MyNodeClass<Key, Value> root;

public MyBinarySearchTree(){
    root = null;
}

@Override
public boolean isEmpty() {
    return root == null;
}

@Override
public Value insert(Key key, Value val) {
    MyNodeClass<Key,Value> newNode = new MyNodeClass<Key,Value>(key,val);
    newNode.setKey(key);
    newNode.setValue(val);
    if(root==null){
        root = newNode;
        return(newNode.getValue());
    }
    else{
        MyNodeClass<Key,Value> current = newNode;
        MyNodeClass<Key,Value> parent;
        while(true){
            {
                parent = current;
                if(current.compareTo(key) == 1)
                {
                    current = current.getLeft();
                    if(current == null)
                    {
                        parent.setLeft(newNode);
                        return parent.getLeft().getValue();
                    }
                }

                else if(current.compareTo(key) == -1){
                    current = current.getRight();
                    if(current == null)
                    {
                        parent.setRight(newNode);
                        return parent.getRight().getValue();
                    }
                }

               else{
                    if(current.compareTo(key) == 0){
                        current.setKey(key);
                        current.setValue(val);
                        return current.getValue();
                    }
                }
            }
        }
    }
}

@Override
public Value find(Key key) {
    MyNodeClass<Key, Value> current = root;
    while (current.compareTo(key) != 0)
    {
        if (current.compareTo(key) == 1)
        {
            current = current.getLeft();
        } else {
            current = current.getRight();
        }
        if(current == null)
            return null;
    }
    return current.getValue();
}

@Override
public Value delete(Key key) {
    MyNodeClass<Key,Value> current = root;
    MyNodeClass<Key,Value> parent = root;
    boolean isLeftChild = true;

    while(current.compareTo(key) != 0) {
        parent = current;
        if (current.compareTo(key) == 1) {
            isLeftChild = true;
            current = current.getLeft();
        } else {
            isLeftChild = false;
            current = current.getRight();
        }
        if(current == null)
            return null;
    }

    if(current.getLeft() == null && current.getRight() == null) {
        if (current == root) {
            root = null;
        } else if (isLeftChild) {
            parent.setLeft(null);
        } else{
            parent.setRight(null);
        }
        return current.getValue();
    }

    else if(current.getRight() == null)
    {
        if(current == root) {
            root = current.getLeft();
        }
        else if(isLeftChild) {
            parent.setLeft(current.getLeft());
        }
        else{
            parent.setRight(current.getLeft());
        }
        return current.getValue();
    }

    else if(current.getLeft() == null)
    {
        if(current == root)
            root = current.getRight();
        else if(isLeftChild)
            parent.setLeft(current.getRight());
        else
            parent.setRight(current.getRight());
        return current.getValue();
    }

    else
    {
        MyNodeClass<Key,Value> successor = getSuccessor(current);

        if(current == root)
            root = successor;
        else if(isLeftChild)
            parent.setLeft(successor);
        else
            parent.setRight(successor);
        successor.setLeft(current.getLeft());
        return current.getValue();
    }
}


@Override
public String stringLevelOrder() {
    return(LevelOrder(root));
}


private MyNodeClass<Key,Value> getSuccessor(MyNodeClass<Key,Value> deleteNode)
{
    MyNodeClass<Key,Value> successorParent = deleteNode;
    MyNodeClass<Key,Value> successor = deleteNode;
    MyNodeClass<Key,Value> current = deleteNode.getRight();

    while(current != null)
    {
        successorParent = successor;
        successor = current;
        current = current.getLeft();
    }

    if(successor != deleteNode.getRight())
    {
        successorParent.setLeft(successor.getRight());
        successor.setRight(deleteNode.getRight());
    }
    return successor;
}
public static void main(String[] args)
{
    MyBinarySearchTree<Double, MyStudent> BST = new MyBinarySearchTree<Double, MyStudent>();
    MyStudent myStud1 = new MyStudent();
    MyStudent myStud2 = new MyStudent();
    MyStudent myStud3 = new MyStudent();
    MyStudent myStud4 = new MyStudent();
    MyStudent myStud5 = new MyStudent();

    myStud1.init("Clarise", 1.1);
    myStud2.init("Christopher", 1.2);
    myStud3.init("John", 1.3);
    myStud4.init("Chloe", 1.4);
    myStud5.init("Goo", 1.5);

    System.out.println(BST.insert(myStud1.getGPA(), myStud1));
    System.out.println(BST.insert(myStud2.getGPA(), myStud2));
    System.out.println(BST.insert(myStud3.getGPA(), myStud3));
    System.out.println(BST.insert(myStud4.getGPA(), myStud4));
    System.out.println(BST.insert(myStud5.getGPA(), myStud5));

    System.out.println("Delete Key 1.0: " +BST.delete(1.3));
    System.out.println("Delete Key 1.4: " +BST.delete(1.4));
    System.out.println("Is Empty?: " +BST.isEmpty());
    System.out.print("Find 3.9: "+  BST.find(3.9));
}

}

主要结果如下:

{Clarise:1.1}

{克里斯托弗:1.2}

{约翰:1.3}

{小克:1.4}

{咕:1.5}

删除密钥1.0:null

删除密钥1.4 null

是空的吗?:错误

查找3.9:null

我不完全确定问题是什么,我得到了其他人的帮助,但他们找不到问题。希望别人能看到我们看不到的东西。

2 个答案:

答案 0 :(得分:0)

insert方法中,在您检查root不是null后,您正在newNode插入newNode而不是root将其插入MyNodeClass<Key,Value> current = newNode; MyNodeClass<Key,Value> parent;

此:

MyNodeClass<Key,Value> current = root;
MyNodeClass<Key,Value> parent;

应该是:

insert

P.S。任何类型的测试都会在一分钟内向您显示size()的问题。你可以写一个toString()方法并看到你的插入不工作,你可以写一个{{1}}方法并查看树的状态,最后你可以调试。

答案 1 :(得分:0)

MyNodeClass<Key,Value> current = newNode;

这里有问题。 如果树不为空,那么“当前”应该从根节点开始。你做的是将“newNode”插入“newNode”。

将代码更改为:

MyNodeClass<Key,Value> current = root;