0

GPOINTER_TO_INT 和 GINT_TO_POINTER 转换为普通 CI 得到:

#include <stdio.h>
#include <stdlib.h>

void *fn(void *v)
{
    int *i;

    i = malloc(sizeof(int));
    *i = (int)(long)v;
    return i;
}

int main(void)
{
    int *i = fn((void *)(long)10);

    printf("%d\n", *i);
    free(i);
    return 0;
}

它是便携式的吗?

为什么演员表long

在 GTK 中用于回调,例如:

#define FLAG 10

static void panel_set_handler(GtkWidget *widget, gpointer data)
{
    panel_set(GPOINTER_TO_INT(data));
}

g_signal_connect(G_OBJECT(menu_item), "activate", G_CALLBACK(panel_set_handler), GINT_TO_POINTER(FLAG));
4

1 回答 1

2

解决方案 1:强制转换

最简单的, Just cast to and from intptr_t。这是 GLib 宏所做的正确版本。

解决方案2:使用堆

如果你有比 intptr_t 更大的东西,或者当你对大小没有信心时,你可以使用动态分配的内存指针,这次没有强制转换:

void* ToHeap(void const *data, size_t dataSize)
{
    void* ret = malloc(dataSize);
    if(ret != NULL)
        memcpy(ret, data, dataSize);
    return ret;
}

int FromHeap(void* heapPtr, void *data, size_t dataSize)
{
    int ret = 0;
    if(heapPtr != NULL)
    {
        memcpy(data, heapPtr, dataSize);
        free(heapPtr);
        ret = 1;
    }
    return ret;
}

这是与 s 一起使用的包装器int

void* IntToHeap(int i)
{
    return ToHeap(&i, sizeof(int));
}
int IntFromHeap(void*heapPtr, int defaultValue)
{
    int ret;
    if(!FromHeap(heapPtr, &ret, sizeof(int))
        ret = defaultValue;
    return ret;
}

你可以这样使用它:

#define FLAG 10

static void panel_set_handler(GtkWidget *widget, gpointer data)
{
    panel_set(IntFromHeap(data, 0));
}

g_signal_connect(G_OBJECT(menu_item), "activate", G_CALLBACK(panel_set_handler), IntToHeap(FLAG));

这种方式有点像你的帖子,减去所有这些演员。

于 2013-06-27T12:26:39.970 回答