Почему функция поиска в двоичном дереве не возвращает адрес root? - PullRequest
0 голосов
/ 26 апреля 2020

Я прошу прощения за то, что слишком много спрашиваю, так что это почти то же самое с моими предыдущими вопросами.
У меня есть балансировочный двоичный код дерева с функцией удаления, поэтому проблема в моей функции searchdelete (). Я не знаю, почему он не вернет root назад. Так, например,

5, 4, 3, 2, 1

И затем я ввожу в меню «Данные, которые должны быть удалены: 1»
, оно переходит к searchdelete (); функция, а затем он успешно
показывает, что данные найдены, found = 1
, но после этого он просто не вернет адрес root. Я понятия не имею, почему это не вернется.
Так в Delete (); функция, она не будет печатать printf("root after = %d", root->data)

#include<stdio.h>
#include<stdlib.h>
#include<malloc.h>
#include<conio.h>

struct node{

    int data, balance;

    struct node *left, *right;

};

int insert(struct node **root, struct node **curr, int data){

    struct node *newNode = (struct node*)malloc(sizeof(struct node));
    newNode -> data = data;
    newNode -> left = NULL;
    newNode -> right = NULL;
    newNode -> balance = 0;

    if((*root) == NULL){
        (*root) = (*curr) = newNode;
        (*root) -> left = NULL;
        (*root) -> right = NULL;
        return 0;
    } else {
        if((*curr)->left == NULL && (*curr)->balance == 0){
            (*curr) -> balance = (*curr) -> balance - 1;
            (*curr) -> left = newNode;
            return 0;
        } else if ((*curr)->right == NULL && (*curr)->balance == -1){
            (*curr) -> balance = (*curr) -> balance + 1;
            (*curr) -> right = newNode;
            return 0;
        } else if ((*curr)->balance == 0 && (*curr)->left->balance == 0){
            (*curr) -> balance = (*curr) -> balance - 1;
            (*curr) = (*curr)->left;
            return insert(root,curr,data);
        } else if ((*curr)->balance < 0 && (*curr)->left->balance < 0){
            (*curr) -> balance = (*curr) -> balance - 1;
            (*curr) = (*curr) -> left;
            return insert(root,curr,data);
        } else if ((*curr)->balance < 0 && (*curr)->left->balance == 0){
            (*curr) -> balance = (*curr) -> balance + 1;
            (*curr) = (*curr)->right;
            return insert(root, curr, data);
        }
    }
}

void preorder(struct node *root){

    if(root == NULL) return;
    printf("%d ", root->data);
    preorder(root->left);
    preorder(root->right);

}

void postorder(struct node *root){

    if(root == NULL) return;
    postorder(root->left);
    postorder(root->right);
    printf("%d ", root->data);

}

void inorder(struct node *root){

    if(root == NULL) return;
    inorder(root->left);
    printf("%d ", root->data);
    inorder(root->right);

}


void search(struct node *root, int *key, int *found){

    if(root == NULL) return;
    search(root->left, key, found);
    if(root->data == *key){
        *found = 1;
        return ;
    }
    search(root->right, key, found);

}

struct node *findMin(struct node *root){

    while(root->left != NULL) root = root->left;
    return root;
}

struct node *searchdelete(struct node *root, int data){

    if(root == NULL) return root;                                    //This is the searchdelete
    searchdelete(root->left, data);
    if(root->data == data){
        printf("found = %d", root->data);
        return root;
    }
    searchdelete(root->right, data);

}

struct node *Delete(struct node *root, int data){

    printf("root before = %d\n", root->data);
    if(root == NULL) return root;
    else if(data != root->data) {
        root = searchdelete(root, data);
        printf("root after = %d\n", root->data);           //this is where it won't print
        system("pause");
    }
    else{
        //Case 1: no child / leaf node
        if(root->left == NULL && root->right == NULL){
            printf("NULL\n");
            free(root);
            root = NULL;
        }
        //Case 2: one child, left or right
        else if(root->left == NULL){
            printf("left null\n");
            struct node *temp = root;
            root = root->right;
            free(temp);
        } else if (root->right == NULL){
            printf("right null\n");
            struct node *temp = root;
            root = root->left;
            free(temp);
        }
        //Case 3: two children
        else{
            printf("two children \n");
            if(root->right->data > root->data){
                struct node *temp = root;
                root = root->right;
                free(temp);
            } else {
                struct node *temp = root;
                root = root->left;
                free(temp);
            }
        }
    }
    system("pause");
    return root;
}


int main(){

    struct node *root, *curr;
    int choice, data, key, found, delKey;
    root = curr = NULL;

    while(1){
        found = 0;
        printf("Balanced Binary Tree Menu\n");
        printf("1. Insert Data\n");
        printf("2. View on pre order\n");
        printf("3. View on post order\n");
        printf("4. View on in order\n");
        printf("5. Search\n");
        printf("6. Delete\n");
        printf("7. Exit\n");
        printf("Pilihan: ");scanf("%d", &choice);fflush(stdin);

        if(choice == 1){
            printf("Enter data : "); scanf("%d", &data);
            curr = root;
            insert(&root, &curr, data);
        } else if (choice == 2){
            preorder(root);
            system("pause");
        } else if (choice == 3){
            postorder(root);
            system("pause");
        } else if (choice == 4){
            inorder(root);
            system("pause");
        } else if (choice == 5){
            printf("Search: "); scanf("%d", &key);
            search(root, &key, &found);
            if(found == 1){
                printf("Data found !\n");
            } else {
                printf("Data not found !\n");
            }
            system("pause");
        } else if (choice == 6){
            printf("Enter data to be deleted: "); scanf("%d", &delKey);
            Delete(root, delKey);
        } else if (choice == 7){
            return 1;
        }
        system("cls");
    }

    return 0;
}


1 Ответ

1 голос
/ 26 апреля 2020

Почему моя функция поиска в двоичном дереве не возвращает адрес root?

Я понятия не имею, почему он не вернется.

, поскольку отсутствуют 2 return

Может быть:

struct node * searchdelete(struct node *root, int data){
    if(root == NULL)
      return root;
    if(root->data == data){
        printf("found = %d", root->data); /* debug, to be removed */
        return root;
    }
    if ((root = searchdelete(root->left, data)) != NULL)
      return root;
    return searchdelete(root->right, data);
}

В поиск почему ключ int* вместо int похоже на searchdelete ? Для меня функция search бесполезна, searchdelete уже достаточно, чтобы узнать, присутствует ли ключ в дереве, просто вызывая его и сравнивая возвращаемое значение с NULL. Поэтому лучше удалить search и переименовать searchdelete в search (или find )

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

Нет никакого интереса иметь двоичное дерево, если элементы случайным образом размещаются в


Ваше определение Удалить имеет много проблем, включая неопределенное поведение, поскольку вы освобождаете ячейки, не удаляя их из дерева

...