- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我需要帮助来理解从 malloc
获得的调试消息。我编写了一个函数,通过递归遍历 trie 上的该字符串并释放未被另一个字符串使用的所有内容,从 trie 中删除指定的字符串。这涉及到下到最后一个节点并释放它,然后返回堆栈并检查每个级别以查看该级别是否也未被其他字符串使用,如果是,则释放它们。一旦它到达第一个被其他东西使用的,它就会停止。
当我只删除一个字符串时,该函数似乎工作正常,但当我删除第二个字符串时,我开始遇到问题。到目前为止,这是我的程序的完整代码(请注意,其中一些行用于测试/调试目的,而不是程序的组成部分):
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
struct node {
char chr;
bool end;
struct node *children[128];
};
void add( struct node *, char * );
void del( struct node *, char * );
bool isMember( struct node *, char * );
bool recursiveDel( struct node *, char * );
// del is really just a dummy function that calls recursiveDel.
int main( int argc, char **argv ){
struct node *trie = (struct node *) malloc( sizeof( struct node ) );
for( int i = 1; i < argc; i++ ){
add( trie, argv[i] );
}
del( trie, argv[1] );
del( trie, argv[2] );
for( int i = 1; i < argc; i++ ){
printf( "%d\n", isMember( trie, argv[i] ) );
}
return 0;
}
void add( struct node *trie, char *str ){
int i = 0;
while( str[i] ){
// Check/goto next node
// If NULL, create next node
if( trie->children[str[i]] == NULL )
trie->children[str[i]] = (struct node *) malloc( sizeof( struct node ) );
trie = trie->children[str[i++]];
}
trie->end = true;
}
void del( struct node *trie, char *str ){
if( isMember( trie, str ) ){
recursiveDel( trie, str );
}
}
bool isMember( struct node *trie, char *str ){
int i = 0;
struct node *cur = trie;
while( str[i] ){
if( trie->children[str[i]] == NULL ) return false;
else trie = trie->children[str[i++]];
}
return trie->end;
}
// Features of this function:
// When it gets to the leaf, it deletes that node and then starts going back up the call stack
// Each call passes a Boolean value back up the call stack.
// This boolean value indicates whether or not the node was deleted.
// If the value returned from the lower node is true, then that means check the next node up to see if it should be deleted.
// If false do nothing, because there are other strings using this node.
bool recursiveDel( struct node *trie, char *str ){
printf( "%p, %d, %s\n", trie, trie->end, str );
if( trie->end ){
free( trie );
return true;
}
bool deleted = recursiveDel( trie->children[str[0]], str+1 );
if( deleted ){
int used = 0;
// Loop checks to see if the node
// is used by any other strings.
for( int i = 0; i < 128; i++ ){
if( trie->children[i] ){
used++;
break;
}
}
if( used <= 1 ){
free( trie );
return true;
}
}
return false;
}
问题似乎出现在这个 block 中,我试图释放字符串的终止节点:
if( trie->end ){
free( trie );
return true;
}
我收到一条消息说该节点不存在,因此无法释放...
bash-3.2$ ./trie hello world
0x7fd370802000, 0, hello
0x7fd370800600, 0, ello
0x7fd370802600, 0, llo
0x7fd370802c00, 0, lo
0x7fd370803200, 0, o
0x7fd370803800, 1,
0x7fd370802000, 0, world
0x7fd370803e00, 0, orld
0x7fd370804400, 0, rld
0x7fd370804a00, 0, ld
0x7fd370805000, 0, d
0x7fd370805600, 1,
trie(43216,0x7fff7818e000) malloc: *** error for object 0x7fd370802000: pointer being freed was not allocated
*** set a breakpoint in malloc_error_break to debug
Abort trap: 6
bash-3.2$
似乎是这样的情况,当我尝试删除第二个字符串时,它会按预期转到最后一个节点,但它不会返回堆栈,而是继续并尝试释放之后的下一个节点,显然它不能这样做,因为这是一个叶节点。
这似乎是未定义的行为,但与此同时,程序会以一种非常可预测的方式失败——第一次字符串删除总是成功,而第二次字符串删除总是不成功。我对此一头雾水。
此外,malloc
失败时给出的地址似乎有点不对劲。最后几个地址都相差0x600
,但是这个地址和最后一个地址相差0xa00
。我知道堆内存分配是不可预测的,但我只是想指出这一点。
比这更奇怪的是 malloc
给出的地址与最后打印的地址不同,尽管失败的 free
操作紧跟在最后一个之后打印
。这似乎表明编译器正在 printf
行和 if(trie->end) free(trie)
部分之间插入一个指针前进操作。常识表明这是荒谬的,但我不知道还有什么其他解释。
最佳答案
代码的相关部分:
int main( int argc, char **argv ){
struct node *trie = (struct node *) malloc( sizeof( struct node ) );
for( int i = 1; i < argc; i++ ){
add( trie, argv[i] );
}
…
}
void add( struct node *trie, char *str ){
int i = 0;
while( str[i] ){
if( trie->children[str[i]] == NULL )
// ^^^^^^^^^^^^^^^^^^^^^^
您正在分配一个struct node
,然后访问它的.children[...]
指针而不初始化它。未定义的行为。
您需要在分配后初始化您的节点
。
关于c - 为什么释放 trie 中的字符串会导致 malloc 错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49959859/
我想做的是让 JTextPane 在 JPanel 中占用尽可能多的空间。对于我使用的 UpdateInfoPanel: public class UpdateInfoPanel extends JP
我在 JPanel 中有一个 JTextArea,我想将其与 JScrollPane 一起使用。我正在使用 GridBagLayout。当我运行它时,框架似乎为 JScrollPane 腾出了空间,但
我想在 xcode 中实现以下功能。 我有一个 View Controller 。在这个 UIViewController 中,我有一个 UITabBar。它们下面是一个 UIView。将 UITab
有谁知道Firebird 2.5有没有类似于SQL中“STUFF”函数的功能? 我有一个包含父用户记录的表,另一个表包含与父相关的子用户记录。我希望能够提取用户拥有的“ROLES”的逗号分隔字符串,而
我想使用 JSON 作为 mirth channel 的输入和输出,例如详细信息保存在数据库中或创建 HL7 消息。 简而言之,输入为 JSON 解析它并输出为任何格式。 最佳答案 var objec
通常我会使用 R 并执行 merge.by,但这个文件似乎太大了,部门中的任何一台计算机都无法处理它! (任何从事遗传学工作的人的附加信息)本质上,插补似乎删除了 snp ID 的 rs 数字,我只剩
我有一个以前可能被问过的问题,但我很难找到正确的描述。我希望有人能帮助我。 在下面的代码中,我设置了varprice,我想添加javascript变量accu_id以通过rails在我的数据库中查找记
我有一个简单的 SVG 文件,在 Firefox 中可以正常查看 - 它的一些包装文本使用 foreignObject 包含一些 HTML - 文本包装在 div 中:
所以我正在为学校编写一个 Ruby 程序,如果某个值是 1 或 3,则将 bool 值更改为 true,如果是 0 或 2,则更改为 false。由于我有 Java 背景,所以我认为这段代码应该有效:
我做了什么: 我在这些账户之间创建了 VPC 对等连接 互联网网关也连接到每个 VPC 还配置了路由表(以允许来自双方的流量) 情况1: 当这两个 VPC 在同一个账户中时,我成功测试了从另一个 La
我有一个名为 contacts 的表: user_id contact_id 10294 10295 10294 10293 10293 10294 102
我正在使用 Magento 中的新模板。为避免重复代码,我想为每个产品预览使用相同的子模板。 特别是我做了这样一个展示: $products = Mage::getModel('catalog/pro
“for”是否总是检查协议(protocol)中定义的每个函数中第一个参数的类型? 编辑(改写): 当协议(protocol)方法只有一个参数时,根据该单个参数的类型(直接或任意)找到实现。当协议(p
我想从我的 PHP 代码中调用 JavaScript 函数。我通过使用以下方法实现了这一点: echo ' drawChart($id); '; 这工作正常,但我想从我的 PHP 代码中获取数据,我使
这个问题已经有答案了: Event binding on dynamically created elements? (23 个回答) 已关闭 5 年前。 我有一个动态表单,我想在其中附加一些其他 h
我正在尝试找到一种解决方案,以在 componentDidMount 中的映射项上使用 setState。 我正在使用 GraphQL连同 Gatsby返回许多 data 项目,但要求在特定的 pat
我在 ScrollView 中有一个 View 。只要用户按住该 View ,我想每 80 毫秒调用一次方法。这是我已经实现的: final Runnable vibrate = new Runnab
我用 jni 开发了一个 android 应用程序。我在 GetStringUTFChars 的 dvmDecodeIndirectRef 中得到了一个 dvmabort。我只中止了一次。 为什么会这
当我到达我的 Activity 时,我调用 FragmentPagerAdapter 来处理我的不同选项卡。在我的一个选项卡中,我想显示一个 RecyclerView,但他从未出现过,有了断点,我看到
当我按下 Activity 中的按钮时,会弹出一个 DialogFragment。在对话框 fragment 中,有一个看起来像普通 ListView 的 RecyclerView。 我想要的行为是当
我是一名优秀的程序员,十分优秀!