Glib - g_double_hash вызывает g_hash_table_contains - неопределенное поведение - PullRequest
0 голосов
/ 11 марта 2019
typedef gchar* String;
typedef gboolean Boolean;
struct ddata
{
    int typ;
    union{
        gchar* a;
    };         
};
typedef struct ddata* ddata;

Пример структуры, экземпляры которой вставлены в таблицу.

Boolean equals_ddata(ddata a, ddata b)
{
    if (a == NULL || b == NULL)
        {
            printf("\nNULL\n");
            return 0;
        }
    if (a->typ == b->typ)
    {
    switch(a->typ)
    {
        case 1:
            {
            if(strcmp(a->a, b->a) == 0)

                return TRUE;
            break;
        }
        default:
            return TRUE;
    }
}
return FALSE;
}
Boolean hm_equals_ddata(gconstpointer a, gconstpointer b) {
    if (((ddata)a) == NULL || ((ddata)b) == NULL) {
        printf("Key for comparison is NULL in hm_equals_ddata");
        return 0;
    }

return equals_ddata((ddata)a, (ddata)b);
}

int main()
{
    ddata obj1 = (ddata)malloc(sizeof(struct ddata));
    ddata obj2 = (ddata)malloc(sizeof(struct ddata));
    ddata obj3 = (ddata)malloc(sizeof(struct ddata));
obj1->typ = 1;
obj1->a = g_strdup("vijay");
obj2->typ = 1;
obj2->a = g_strdup("vijay");
obj3->typ = 2;
obj3->a = g_strdup("kumar");
printf("\n%s, %s, %s\n", obj1->a, obj2->a, obj3->a);
GHashTable* c = g_hash_table_new(g_double_hash, hm_equals_ddata);
printf("%d\n", g_hash_table_insert(c, obj1, NULL));
printf("%d\n", g_hash_table_insert(c, obj2, NULL));
printf("%d\n", g_hash_table_insert(c, obj3, NULL));
printf("%d\n", g_hash_table_contains(c, obj3));
return 0;
}

Здесь g_hash_table_contains иногда не работает должным образом.Функция равенства, присоединенная при создании хеш-таблицы, не вызывается g_hash_table_contains, это связано с функцией g_double_hash.Так может кто-нибудь объяснить это неопределенное beevour.

1 Ответ

0 голосов
/ 11 марта 2019

Хеш-функции и функции равенства применяются к ключу в записи хеш-таблицы, а не к значению.Ваш код вставляет ключи типа ddata (и значения NULL), но затем пытается хэшировать их как двойные числа и сравнивать ключи на равенство как ddata s.

Попробуйте вместо этого использовать g_direct_hashg_double_hash.

...