Handle-with-cache.c -
// handle-with-cache.c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <glib.h> // Using GLib's hash table for simplicity typedef struct { int user_id; char *name; char *email; // ... other data } UserProfile;
GHashTableIter iter; gpointer key, value; g_hash_table_iter_init(&iter, handle_cache); while (g_hash_table_iter_next(&iter, &key, &value)) { CacheEntry *entry = value; if (entry->ref_count == 0 && (now - entry->last_access) > max_age_seconds) { to_remove = g_list_prepend(to_remove, key); } } handle-with-cache.c
The module handle-with-cache.c exemplifies a classic design pattern: the . A "handle" is an opaque pointer or identifier to a resource, and the cache stores recently accessed handles to avoid redundant initialization or I/O operations. // handle-with-cache
// The cache itself (often a global or passed context) static GHashTable *handle_cache = NULL; static pthread_mutex_t cache_lock = PTHREAD_MUTEX_INITIALIZER; This function does the actual heavy lifting – creating a handle from scratch. // The cache itself (often a global or
// Find the entry for this profile (simplified; real code needs reverse mapping) GHashTableIter iter; gpointer key, value; g_hash_table_iter_init(&iter, handle_cache); while (g_hash_table_iter_next(&iter, &key, &value)) { CacheEntry *entry = value; if (entry->profile == profile) { entry->ref_count--; if (entry->ref_count == 0) { // Last reference - we could evict immediately or mark as stale printf("No more references to user %d, marking for eviction\n", *(int*)key); } break; } }
A common optimization is or using a per-key mutex:
// Store in cache (use user_id as key) int *key = malloc(sizeof(int)); *key = user_id; g_hash_table_insert(handle_cache, key, new_entry);