2016-04-10 1 views
1

Я считаю, что моя функция вставки правильная, но похоже, что новый узел не вставлен в дерево. Я не мог понять, где ошибка. Я ценю любую помощь, спасибо.Невозможно вставить новый узел в двоичное дерево

Существует декларация узла и дерева:

class Node{ 
    int key; 
    Node *right, *left; 
} 

class Tree{ 
public: 
     int init(); 
     Node *root; 
     Node *insert(int key, Node *p); 
}; 

есть функции:

int Tree::init(){ 
    this->root = NULL; return 1; 
} 

Node *Tree::insert(int key, Node *p){ 
    if(p == NULL){ 
    Node *novo = new Node(); 
    novo->key = key; 
    novo->left = NULL; 
    novo->right = NULL; 
    p = novo; 
    } 
    else if(key < p->key){ p->left = insert(key, p->left); } 
    else if(key > p->key){ p->right = insert(key, p->right); } 
    else{ cout << "Error: key already exist" << endl; } 

return p; 
} 

Когда я вызываю функцию в основном, это выглядит, как он не связывает новый Узел

int main() { 
    Tree dictionary; 

    cout << "enter the key"; cin >> key; 

    dictionary.insert(key, dictionary.root); 

    cout << dictionary.root->key; 
} 
+0

Отключить тему: почему 'int Tree :: init()' вместо конструктора? – user4581301

+0

Ваш 'MCVE' не компилируется - я попробовал его с помощью' gcc'. –

ответ

1

В вашей функции insert(), когда дерево пуст или вы достигли последнего узла, вы создаете т.е новый узел:

if(p == NULL){ 
    Node *novo = new Node(); 
    novo->key = key; 
    novo->left = NULL; 
    novo->right = NULL; 
    p = novo;    // ouch !!!! 
    } 

К сожалению, заявление p=novo только обновляет локальный параметр p вашей функции. Его значение исчезнет, ​​как только вы вернетесь из функции. Он не будет обновлять указатель, который вы назвали своей функцией. Таким образом, корень вашего дерева остается NULL (или указатель слева/справа последнего узла).

Чтобы получить эффект, который вы ожидаете (т.е. вашего p распайки обновляет указатель корневого или указатель влево/вправо от последнего узла), вы должны изменить подпись:

Node *insert(int key, Node *& p); // p is passed by reference 

Это будет передать указатель p по ссылке. Затем изменение p будет иметь эффект изменения указателя, который вы использовали для вызова функции, и выдержит длительный эффект вставки.

+0

Вы забыли также упомянуть, что он передает неинициализированную переменную 'dictionary.root' в' main', и она никогда не сможет работать. –

+0

Спасибо, Кристоф !!! Это упражнение, но я не заметил этой детали. –

+0

@AlBundy да, это потому, что OP забыл вызвать init(). Создание конструктора вместо init() позволило бы избежать таких неприятных ошибок ;-) – Christophe