プレフィックスツリーでメモリを扱う方法は? (ANSI C)
-
04-10-2019 - |
質問
私はdict_free()関数でメモリを扱おうとしましたが、それは機能せず、理由はありません。私は何かが足りませんか?何が悪いのかわからない。
編集:dict_free()でfree()を呼び出すと、nullがnullにポインターがポイントしたことがわかりますが、それは起こっていません。
これが私のコードです:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct Dict
{
struct Dict *branches[256];
int index;
}Dict;
void dict_insert_depth(unsigned char*,Dict *,int);
void dict_insert(unsigned char*,Dict *);
void dict_free(Dict *d)
{
if(d!=NULL){
int i;
for(i=0; i<256; i++){
if(d->branches[i] != NULL){
dict_free(d->branches[i]);
free(d->branches[i]);
printf("Is it free?? %s\n",d==NULL?"yes":"no");
}
}
}
}
/**
* Insert word into dictionaR
*/
void dict_insert(unsigned char *w, Dict *d)
{
dict_insert_depth(w,d,0);
}
void dict_insert_depth(unsigned char *w, Dict *d, int depth)
{
if(strlen(w) > depth){
int ch = w[depth];
if(d->branches[ch]==NULL){
d->branches[ch] = malloc(sizeof(struct Dict));
dict_insert_depth(w,d->branches[ch],depth+1);
}else{
dict_insert_depth(w,d->branches[ch],depth+1);
}
}
}
/**
* Check whether a word exists in the dictionary
* @param w Word to be checked
* @param d Full dictionary
* @return If found return 1, otherwise 0
*/
int in_dict(unsigned char *w, Dict *d)
{
return in_dict_depth(w,d,0);
}
int in_dict_depth(unsigned char *w, Dict *d, int depth)
{
if(strlen(w)>depth){
int ch = w[depth];
if(d->branches[ch]){
return in_dict_depth(w, d->branches[ch], depth+1);
}else{
return 0;
}
}else{
return 1;
}
}
解決
ルートノードを解放できない場合を除き、無料のコードは問題ありません。
自由度のテストは間違っています。 free
変数を設定しません NULL
. 。多くの場合、それを明示的に行うことをお勧めしますので、すでに解放されたメモリを読んではいけません。
free(d->branches[i]);
d->branches[i] = NULL; // clobber pointer to freed memory
ルートノードの問題を処理し、おそらくややきれいにするために、これを行います。
void dict_free(Dict *d)
{
if(d!=NULL){
int i;
for(i=0; i<256; i++){
if(d->branches[i] != NULL){
dict_free(d->branches[i]);
d->branches[i] = NULL;
}
}
free(d);
}
}
他のヒント
dict_free(d->branches[i]);
free(d->branches[i]);
printf("Is it free?? %s\n",d==NULL?"yes":"no");
これはチェックします D、 しかし、あなたは変更しません d ループの中。あなたがそれをチェックしているので d 上記のヌルではありませんが、これは常にノーを印刷します。
void dict_free(Dict* d) {
if (d) {
for(int i = 0; i < 256; i++) {
if (d->branches[i]) {
dict_free(d->branches[i]);
free(d->branches[i]);
d->branches[i] = 0; // mark this branch as freed
// important if d is reused, and since dict_free doesn't
// free(d), it could be
}
}
}
}
私はあなたの既存のコードを解放しないということに従いました D、 ただし、物事を変更して、DICTが常に同じ方法で割り当てられます(例:DICT_NEW関数を追加)dict_freeも通過したオブジェクトを解放します。
所属していません StackOverflow