questão tipo Java genérico binário Pesquisa Árvore
-
03-07-2019 - |
Pergunta
Eu estou trabalhando neste trabalho de casa que é tipo de confundir-me ...
Estou fornecido com a seguinte classe BinarySearchTree
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);
}
}
=============================================== =================================
Agora, eu tenho um outro estudante classe .... Eu quero criar uma árvore de busca binária de Student objetos ..
BinarySearchTree<Student> tree = new BinarySearchTree<Student>();
No entanto, quando eu faço que eu recebo o seguinte erro:
incompatibilidade Limite: O tipo de estudante não é um substituto válido para o parâmetro limitada> do tipo BinarySearchTree
Todas as idéias que está acontecendo aqui ... Eu não posso descobrir isso.
Solução
public class BinarySearchTree<T extends Comparable<T>>
Um argumento formal genéricos, no seu caso T, listas do que é necessário para uma classe para ser um T. válida no seu caso, você disse, "ser um t válido, uma classe deve implementar Comparável" (A palavra-chave é "estende", mas na prática, que significa "estende ou implementa".)
Em sua instanciação, T é estudante. Se substituirmos Student para T:
public class BinarySearchTree<Student extends Comparable<Student>>
é que uma afirmação verdadeira? O estudante realmente implementar comparáveis?
Se isso acontecer, estudante se encaixa a exigência de ser um T, e assim você pode usar Student como o parâmetro real para o parâmetro formal T.
Se não, você começa a reclamação do compilador que você viu.
Na verdade, para cobrir situações mais complicadas em que a implementação de uma subclasse de Comparável é feito por uma classe super, a forma mais geral seria:
public class BinarySearchTree<T extends Comparable<? super T > >
Então, você precisa fazer Student implementar Comparable
Note que eu não dizer que o compilador está procurando um Student.compareTo
. Ele nem sequer chegar tão longe. É olhando para ver se T (no seu caso, Student) é declarada como implementar Comparable
Agora acrescentando implements Comparable< Student >
para Student irá também fazer o compilador garantir que não há um método public int compareTo
em Student. Mas sem os "instrumentos comparáveis", mesmo que o compilador sabe que há um método Student.compareTo
, ele não sabe que que compareTo
é o Comparable.compareTo
.
(Em outras palavras, nós estamos olhando para a implementação declarou, não apenas que não passa a ser um método com o nome ea assinatura direita.)
Outras dicas
A classe Student implementar comparáveis?
mas não tenho certeza de como implementar o método compareTo.
Basicamente, é algo como o seguinte. Como a ordenação funciona você tem que decidir.
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
}
}