- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试序列化和反序列化 GHashTable
JSON 和 valgrind
的输入和输出报告说这样做的结果肯定会丢失内存。 g_hash_table_new
有抑制在 glib.suppressions
他们分发的文件,但没有 g_hash_table_new_full
的文件这是我看到的一堆。我的哈希表是 GObject
已使用 g_param_spec_pointer
设置的属性,序列化它的函数是:
static JsonNode *
foo_obj_serialize_property (JsonSerializable *serializable,
const gchar *name,
const GValue *value,
GParamSpec *pspec)
{
JsonNode *retval = NULL;
if (g_strcmp0 (name, "list") == 0)
{
GHashTable *list = NULL;
GHashTableIter iter;
JsonArray *arr = NULL;
gpointer key, val;
retval = json_node_new (JSON_NODE_ARRAY);
g_return_val_if_fail (value != NULL, retval);
g_return_val_if_fail (G_VALUE_HOLDS_POINTER (value), retval);
list = g_value_get_pointer (value);
arr = json_array_new ();
if (list != NULL)
{
g_hash_table_iter_init (&iter, list);
while (g_hash_table_iter_next (&iter, &key, &val))
{
JsonNode *node = NULL;
JsonObject *obj = NULL;
FooItem *item;
item = FOO_ITEM (val);
node = json_gobject_serialize (G_OBJECT (item));
if (JSON_NODE_HOLDS_OBJECT (node))
{
obj = json_node_get_object (node);
json_array_add_object_element (arr, obj);
}
}
}
json_node_take_array (retval, arr);
}
return retval;
}
反序列化是:
static gboolean
foo_obj_deserialize_property (JsonSerializable *serializable,
const gchar *name,
GValue *value,
GParamSpec *pspec,
JsonNode *property_node)
{
gboolean retval = FALSE;
if (g_strcmp0 (name, "list") == 0)
{
GHashTable *list;
JsonArray *arr;
arr = json_node_get_array (property_node);
list = g_hash_table_new_full (g_str_hash,
g_str_equal,
g_free,
NULL);
for (gint i = 0; i < json_array_get_length (arr); i++)
{
g_autoptr (FooItem) item = NULL;
JsonNode *node = NULL;
node = json_array_get_element (arr, i);
item = FOO_ITEM (json_gobject_deserialize (FOO_TYPE_ITEM, node));
g_return_val_if_fail (FOO_IS_ITEM (item), FALSE);
g_object_ref (item);
g_hash_table_insert (list,
g_strdup (foo_item_get_key (item)),
item);
}
g_value_set_pointer (value, list);
retval = TRUE;
}
return retval;
}
我的GObject
类属性获取/设置函数是:
static void
foo_obj_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
FooObj *self = FOO_OBJ (object);
switch (prop_id)
{
case PROP_LIST:
g_value_set_pointer (value, self->list);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
foo_obj_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
FooObj *self = FOO_OBJ (object);
switch (prop_id)
{
case PROP_LIST:
foo_obj_set_list (self, g_value_get_pointer (value));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
最后是类的 getter/setter:
GHashTable *
foo_obj_get_list (FooObj *self)
{
GHashTable *list;
g_return_val_if_fail (FOO_IS_OBJ (self), NULL);
g_object_get (self, "list", &list, NULL);
return list;
}
void
foo_obj_set_list (FooObj *self,
GHashTable *list)
{
g_return_if_fail (FOO_IS_OBJ (self));
if (self->list == list)
return;
if (list)
g_hash_table_ref (list);
if (self->list)
g_hash_table_unref (self->list);
self->list = list;
g_object_notify_by_pspec (G_OBJECT (self), properties [PROP_LIST]);
}
我正在使用 valgrind --tool=memcheck --leak-check=full --show-leak-kinds=all --leak-resolution=high --num-callers=20 --suppressions=../tests/glib.supp foo
运行内存检查,并且我收到序列化内存丢失的消息,显示:
==24303== 32 bytes in 1 blocks are definitely lost in loss record 1,029 of 1,852
==24303== at 0x483877F: malloc (vg_replace_malloc.c:299)
==24303== by 0x494CAB1: g_malloc (in /usr/lib/libglib-2.0.so.0.6000.4)
==24303== by 0x492C8E4: g_slice_alloc (in /usr/lib/libglib-2.0.so.0.6000.4)
==24303== by 0x4933839: g_slice_alloc0 (in /usr/lib/libglib-2.0.so.0.6000.4)
==24303== by 0x4D72CEE: json_node_alloc (in /usr/lib/libjson-glib-1.0.so.0.400.4)
==24303== by 0x4D72D37: json_node_new (in /usr/lib/libjson-glib-1.0.so.0.400.4)
==24303== by 0x4D77B37: json_gobject_serialize (in /usr/lib/libjson-glib-1.0.so.0.400.4)
==24303== by 0x4867AC9: foo_item_serialize_property (foo-item.c:71)
==24303== by 0x4D779C1: ??? (in /usr/lib/libjson-glib-1.0.so.0.400.4)
==24303== by 0x4D77B42: json_gobject_serialize (in /usr/lib/libjson-glib-1.0.so.0.400.4)
==24303== by 0x4861681: foo_obj_serialize_property (foo-obj.c:146)
==24303== by 0x4D779C1: ??? (in /usr/lib/libjson-glib-1.0.so.0.400.4)
==24303== by 0x4D77B42: json_gobject_serialize (in /usr/lib/libjson-glib-1.0.so.0.400.4)
==24303== by 0x4D77B82: json_gobject_to_data (in /usr/lib/libjson-glib-1.0.so.0.400.4)
以及反序列化:
==24303== 184 (88 direct, 96 indirect) bytes in 1 blocks are definitely lost in loss record 1,774 of 1,852
==24303== at 0x483877F: malloc (vg_replace_malloc.c:299)
==24303== by 0x494CAB1: g_malloc (in /usr/lib/libglib-2.0.so.0.6000.4)
==24303== by 0x492C8E4: g_slice_alloc (in /usr/lib/libglib-2.0.so.0.6000.4)
==24303== by 0x4966C5E: g_hash_table_new_full (in /usr/lib/libglib-2.0.so.0.6000.4)
==24303== by 0x4861AB2: foo_obj_deserialize_property (foo-obj.c:264)
==24303== by 0x4D77F00: ??? (in /usr/lib/libjson-glib-1.0.so.0.400.4)
==24303== by 0x4D7826B: json_gobject_from_data (in /usr/lib/libjson-glib-1.0.so.0.400.4)
==24303== by 0x4862BEE: foo_obj_deserialize (foo-obj.c:472)
我很难看到我错过了什么,并且各种清除指针的尝试都会导致双重释放或关于引用计数== 0的错误。我知道有很多 GLib
由于它处理内存的方式而受到抑制,但我不知道这些是否属于此范围。
我可能可以通过使用 json-glib
注册一个函数来对哈希表进行装箱并简化序列化,但除非需要,否则我不想朝那个方向发展。
最佳答案
node = json_gobject_serialize (G_OBJECT (item));
在这行之后,您不会对 node
执行任何操作,既不会将其存储在某个地方,也不会释放它。这会导致内存泄漏。另请注意,JsonNode
是 GBoxed
,而不是 GObject
。使用json-node-free释放它。
我还建议您阅读有关引用计数和 analysing valgrind output 的内容。 。
The stack trace tells you where the leaked memory was allocated. Memcheck cannot tell you why the memory leaked, unfortunately.
对于deserialize
方法,它被分配在(foo-obj.c:264),即g_hash_table_new_full
。您创建一个哈希表,但不要销毁它。
关于c - JSON (GLib) 序列化泄漏 GHashTable 的内存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56980385/
我想在 GLib 中将整数值转换为字符串。是否有任何宏或函数?或者我可以在同一个双向链表中存储不同的数据类型吗? 最佳答案 gchar *my_string = g_strdup_printf("%i
所以基本上我正在编写以下步骤来创建基于 dbus 的应用程序。1. g_bus_watch_name(特定服务出现的回调)2. g_signal_connect 附加到服务提供的某些信号。 g_sig
所以基本上我正在编写以下步骤来创建基于 dbus 的应用程序。1. g_bus_watch_name(特定服务出现的回调)2. g_signal_connect 附加到服务提供的某些信号。 g_sig
我尝试使用JUCE框架构建cpp程序,但出现错误 make: *** No rule to make target '/usr/include/glib-2.0/glib/gurifuncs.h',
我有一些使用 dbus-glib 通过 dbus 编写服务器导出和接口(interface)。 我正在尝试编写一个将字节数组作为输入的方法,但是我很难确定在我的方法参数中使用什么类型。 例如,如果我有
我一直在研究 python-mpdor 的源代码它提到它是 gobject-based, for easy event handling (in the high-level client class
假设我有一个 GLib.Array并想将其转换为 Item[] ,我将如何在 Vala 中做到这一点? 最佳答案 首先,除非您需要与现有代码进行互操作,否则不要使用 GLib.Array。使用 GLi
我正在尝试使用 GLib 的 spawn_command_line_sync 将 echo 的输出通过管道传输到命令中方法。我遇到的问题是 echo 将整个命令解释为参数。 为了更好地解释,我在我的代
我似乎无法使用 glib.h 编译这个基本程序... #include glib.h #include stdio.h int main () { return ((glib_major_ver
我刚开始学习 vala。我尝试了来自 vala tutorial 的以下程序. class Demo.Hello : Glib.Object { public static int main( s
在调查一些性能问题时,我最终进入了 gthread-posix.c。 在那里我找到了如下代码: static void __attribute__((noinline)) g_mutex_lock_s
这个问题在这里已经有了答案: Undefined reference to 'pthread_create' — linker command option order (libraries bef
我正在学习使用 GLib 编写简单、高效的套接字服务器的基础知识。我正在试验 GSocketService。到目前为止,我似乎只能接受连接,但它们会立即关闭。从文档中我无法弄清楚我缺少的步骤。我希望有
我在 gobject 上阅读了一个维基百科页面,上面写着, Depending only on GLib and libc, GObject is a cornerstone of GNOME and
我尝试了sudo apt-get install libglib2.0-dev并得到了 libglib2.0-dev is already the newest version. 我仍然收到错误 co
所以我正在尝试使用 GtkSourceViewmm 在 C++ 中使用 GtkSourceView ,其文档和支持水平给我的印象是很长一段时间没有仔细研究它。但我始终是一个乐观主义者 :) 我正在尝试
我是编程新手。我正在创建一个 Qt GUI 应用程序。这是一个小的拼字游戏。问题是每当我在构建它后从命令提示符启动这个程序时,我都会在命令提示符下得到以下输出: JumbleGame 1.0 Laun
我正在尝试创建一个相当简单的使用自动工具的 BitBake 配方,您可以在这里看到: SUMMARY = "an example autotools recipe" LICENSE = "MIT" L
我正在尝试创建一个相当简单的使用自动工具的 BitBake 配方,您可以在这里看到: SUMMARY = "an example autotools recipe" LICENSE = "MIT" L
我正在查看一段代码,它直到最近才有效。基本上,我有一个 C++ 类,我在其中使用 G_LOCK_DEFINE 宏保护一个变量。 class CSomeClass { private: gulo
我是一名优秀的程序员,十分优秀!