以下代码是NCURSES菜单库中的示例。我不确定代码有什么问题,但valgrind报告了一些问题。任何想法......
==4803== 1,049 (72 direct, 977 indirect) bytes in 1 blocks are definitely lost in loss record 25 of 36
==4803== at 0x4C24477: calloc (vg_replace_malloc.c:418)
==4803== by 0x400E93: main (in /home/gerardoj/a.out)
==4803==
==4803== LEAK SUMMARY:
==4803== definitely lost: 72 bytes in 1 blocks
==4803== indirectly lost: 977 bytes in 10 blocks
==4803== possibly lost: 0 bytes in 0 blocks
==4803== still reachable: 64,942 bytes in 262 blocks
源代码:
#include <curses.h>
#include <menu.h>
#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
#define CTRLD 4
char *choices[] = {
"Choice 1",
"Choice 2",
"Choice 3",
"Choice 4",
"Choice 5",
"Choice 6",
"Choice 7",
"Exit",
}
;
int main()
{
ITEM **my_items;
int c;
MENU *my_menu;
int n_choices, i;
ITEM *cur_item;
/* Initialize curses */
initscr();
cbreak();
noecho();
keypad(stdscr, TRUE);
/* Initialize items */
n_choices = ARRAY_SIZE(choices);
my_items = (ITEM **)calloc(n_choices + 1, sizeof(ITEM *));
for (i = 0; i < n_choices; ++i) {
my_items[i] = new_item(choices[i], choices[i]);
}
my_items[n_choices] = (ITEM *)NULL;
my_menu = new_menu((ITEM **)my_items);
/* Make the menu multi valued */
menu_opts_off(my_menu, O_ONEVALUE);
mvprintw(LINES - 3, 0, "Use <SPACE> to select or unselect an item.");
mvprintw(LINES - 2, 0, "<ENTER> to see presently selected items(F1 to Exit)");
post_menu(my_menu);
refresh();
while ((c = getch()) != KEY_F(1)) {
switch (c) {
case KEY_DOWN:
menu_driver(my_menu, REQ_DOWN_ITEM);
break;
case KEY_UP:
menu_driver(my_menu, REQ_UP_ITEM);
break;
case ' ':
menu_driver(my_menu, REQ_TOGGLE_ITEM);
break;
case 10:
{
char temp[200];
ITEM **items;
items = menu_items(my_menu);
temp[0] = '\0';
for (i = 0; i < item_count(my_menu); ++i)
if(item_value(items[i]) == TRUE) {
strcat(temp, item_name(items[i]));
strcat(temp, " ");
}
move(20, 0);
clrtoeol();
mvprintw(20, 0, temp);
refresh();
}
break;
}
}
unpost_menu(menu);
free_item(my_items[0]);
free_item(my_items[1]);
free_item(my_items[2]);
free_item(my_items[3]);
free_item(my_items[4]);
free_item(my_items[5]);
free_item(my_items[6]);
free_item(my_items[7]);
free_menu(my_menu);
endwin();
}
答案 0 :(得分:3)
根据NCURSES Programming Howto,使用菜单库需要执行以下步骤:
初始化curses 使用new_item()创建项目。您可以指定项目的名称和描述。 通过指定要附加的项目,使用new_menu()创建菜单。 使用menu_post()发布菜单并刷新屏幕。 使用循环处理用户请求,并使用menu_driver对菜单进行必要的更新。 使用menu_unpost()取消发布菜单 释放free_menu()分配给菜单的内存 使用free_item()释放分配给项目的内存 结束诅咒
从我的代码中可以看出:
my_items
指针数组。这当然是一个泄密。正如@ lh3所说,使用-g
选项进行编译会让Valgrind给出丢失内存的行号。
编辑(响应您的评论):my_items
是动态分配的动态创建菜单项指针数组。换句话说,你有一个动态内存块,它包含一堆指向动态分配的ncurses结构(菜单项)的指针。所以,一旦你完成清理,你需要释放每个动态分配的ncurses结构,然后你需要释放那些包含这些结构指针的内存块。
换句话说,每个calloc
或malloc
需要一个free
,每个new_item
需要一个free_item
,依此类推。
for (i = 0; i < n_choices; ++i) {
free_item(my_items[i]);
}
free(my_items);
答案 1 :(得分:1)
Valgrind需要注意的事项(经常出现在Valgrind user's mailing list上):
still reachable: 64,942 bytes in 262 blocks
这只是引用在退出时main()
仍然可以访问的块,它们(在任何现代内核下)无论如何都会被操作系统回收。
虽然调用退出前显式free()
每个已分配块的良好做法,但这不是technically
泄漏的内存,因为它仍然可以在退出时到达。
专注于Josh Kelly建议的直接,间接和可能丢失的块。这只是已经指出泄漏可能来源的答案的补充。
答案 2 :(得分:0)
尝试使用--leak-check = full运行valgrind吗?
答案 3 :(得分:0)
free_item(my_items[7]);
free(my_items);