我有以下删除min元素的功能:
int BinaryTree::delete_min_helper(TreeNode *node){
while(node->left != NULL){
node = node->left;
}
if(node == root){
return 1;
}
delete node;
node = NULL;
return 0;
}
我总是将指向根节点的指针传递给此函数。我已经尝试了这个以及它的一些变体,但似乎总是删除错误的东西或者根本不删除任何东西。有什么想法吗?
这是一个可编辑的例子:
#ifndef _TREE_H_
#define _TREE_H_
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
struct TreeNode {
int val;
char *str;
TreeNode *left;
TreeNode *right;
};
class BinaryTree {
public:
BinaryTree();
~BinaryTree();
int insert_node(unsigned int val, char *str);
TreeNode *find_min();
int delete_min();
void print();
private:
int insert_node_helper(TreeNode *&node, unsigned int val, char *str);
int delete_min_helper(TreeNode *node);
void print_helper(TreeNode *node);
TreeNode *root;
};
#endif
BinaryTree::BinaryTree(){
this->root = NULL;
}
BinaryTree::~BinaryTree(){
}
int BinaryTree::insert_node(unsigned int val, char *str){
return insert_node_helper(this->root, val, str);
}
int BinaryTree::insert_node_helper(TreeNode *&node, unsigned int val, char *str){
if(node == NULL){
node = new TreeNode;
node->val = val;
node->str = strdup(str);
node->left = NULL;
node->right = NULL;
if(node != NULL){
return 0;
}else{
puts("inserted null node");
return 1;
}
}else if(val <= node->val){
return insert_node_helper(node->left, val, str);
}else if(val > node->val){
return insert_node_helper(node->right, val, str);
}
return 1;
}
void BinaryTree::print(){
print_helper(this->root);
}
void BinaryTree::print_helper(TreeNode *node){
if(node != NULL){
print_helper(node->right);
printf("%d occurrences of \"%s\"\n", node->val, node->str);
print_helper(node->left);
}
}
TreeNode *BinaryTree::find_min(){
TreeNode *temp = this->root;
while(temp->left != NULL){
temp = temp->left;
}
return temp;
}
int BinaryTree::delete_min(){
return delete_min_helper(root);
}
int BinaryTree::delete_min_helper(TreeNode *node){
while(node->left != NULL){
node = node->left;
}
if(node == root){
puts("attempted to delete root");
return 1;
}
delete node;
node = NULL;
return 0;
}
#include <time.h>
int main(){
BinaryTree bt;
srand(time(NULL));
for(int i = 0; i < 10; i++){
bt.insert_node(rand() % 20, "test");
}
bt.print();
printf("min val = %d\n", bt.find_min()->val);
puts("#################################");
bt.delete_min();
bt.print();
printf("min val = %d\n", bt.find_min()->val);
return 0;
}
答案 0 :(得分:4)
您正在引用指针作为参数(TreeNode *&node
),然后对其进行修改。因此,如果将树的根传递给此方法,最终将根设置为NULL
。
此外,这实际上绝对没有删除任何内容,因为在将指针设置为NULL后调用delete,并且调用空指针的delete
不执行任何操作。
更新:
我认为root
是BinaryTree的成员。如果您发布了Short, Self Contained, Compilable Example,那么帮助您会更容易。
你仍然以root身份作为参考。因此node == root
将始终为真,您永远不会达到delete
声明。所以实际上你并没有删除任何内容,但是你要将你的root更改为最左边的子节点。
尝试将方法签名更改为
int BinaryTree::delete_min_helper(TreeNode *node){
即没有&符号(&
)。
编辑后:
好的,现在我运行你的程序,发现了你的第二个问题。在行
node = NULL;
您正在更改父节点中本地指针的值,而不是left
- 指针的值。这意味着,当您下次遍历树时,您将访问指向您刚刚删除的内存的指针。这会触发未定义的行为。要解决此问题,您应该在删除叶节点时将父节点的left
- 指针设置为0.