c中回调寄存器函数中userdata的用途是什么?

时间:2018-06-15 10:55:46

标签: c callback function-pointers

在寄存器回调函数中有两个参数。一个是函数指针,第二个是userdata

int callback_register(fn_ptr cb, void *userdata);
//fn_ptr is typedef

在回调期间,相同的userdata将作为参数发回。 我理解使用发送函数指针,但不是发送userdata。 任何人都可以告诉我们如何使用它?

1 个答案:

答案 0 :(得分:2)

如果您只有一个指向回调函数的函数指针,则无法将额外数据传递回回调函数。

例如,假设我有一些我想要使用qsort排序的字符串。我可能会从这样一些简单的代码开始:

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

#define Sizeofarray(a) (sizeof(a) / sizeof(*a))

int compar(const void *, const void *);

int main()
{
    char *data[] = { "apple", "pear", "1", "2", "10" };
    int i;

    printf("unsorted:\n");
    for(i = 0; i < Sizeofarray(data); i++) printf("%s\n", data[i]);

    qsort(data, Sizeofarray(data), sizeof(*data), compar);

    printf("\nsorted:\n");
    for(i = 0; i < Sizeofarray(data); i++) printf("%s\n", data[i]);
}

int compar(const void *p1, const void *p2)
{
    const char *s1 = *(const char **)p1;
    const char *s2 = *(const char **)p2;
    return strcmp(s1, s2);
}

这个程序工作正常,打印出来:

unsorted:
apple
pear
1
2
10

sorted:
1
10
2
apple
pear

但这是使用strcmp的字母排序。假设我想要数字排序选项(就像标准的Unix / Linux sort命令及其-n选项一样)。而且,假设我真的想让它成为一个由运行时变量控制的选项。我可以编写一个新的,稍微复杂的比较函数,如下所示:

int compar2(const void *p1, const void *p2)
{
    const char *s1 = *(const char **)p1;
    const char *s2 = *(const char **)p2;

    if(numeric)
         return atoi(s1) > atoi(s2);
    else return strcmp(s1, s2);
}

numeric标志设置为true,输入现在排序为

apple
pear
1
2
10

(首先出现的是atoi&#34;将&#34;它们转换为0。)

但是,重要的问题是, numeric标志来自哪里?如果我只有qsort并且回调函数不带用户可指定的上下文,我别无选择,只能将numeric标志作为全局变量:

int numeric;

int main()
{
    char *data[] = { "apple", "pear", "1", "2", "10" };
    int i;

    printf("unsorted:\n");
    for(i = 0; i < Sizeofarray(data); i++) printf("%s\n", data[i]);

    numeric = 0;
    qsort(data, Sizeofarray(data), sizeof(*data), compar2);

    printf("\nsorted alphabetically:\n");
    for(i = 0; i < Sizeofarray(data); i++) printf("%s\n", data[i]);

    numeric = 1;
    qsort(data, Sizeofarray(data), sizeof(*data), compar2);

    printf("\nsorted numerically:\n");
    for(i = 0; i < Sizeofarray(data); i++) printf("%s\n", data[i]);
}

这也有效。但当然没有人喜欢全球变数。

这就是&#34; userdata&#34;的概念。参数进来了。我不知道它是多么标准,但我的系统提供了一个名为qsort的{​​{1}}变体。 (&#34; qsort_r&#34;用于&#34;可重入&#34;。)对于这个版本,比较函数会传递一个额外的参数,我可以随心所欲地做任何事情。在这里,我可以将它指向我的r标志,现在我的numeric标志不一定是全局变量:

numeric

因此,简而言之,问题的答案&#34;回调函数中userdata参数的用途是什么?&#34;因此,调用函数不必使用全局变量将额外的上下文信息传递给它们的回调函数。&#34;

脚注:在这种情况下,我可以采取不同的方法。我本可以定义两个不同的int compar3(void *, const void *, const void *); int main() { char *data[] = { "apple", "pear", "1", "2", "10" }; int i; int numeric; printf("unsorted:\n"); for(i = 0; i < Sizeofarray(data); i++) printf("%s\n", data[i]); numeric = 0; qsort_r(data, Sizeofarray(data), sizeof(*data), &numeric, compar3); printf("\nsorted alphabetically:\n"); for(i = 0; i < Sizeofarray(data); i++) printf("%s\n", data[i]); numeric = 1; qsort_r(data, Sizeofarray(data), sizeof(*data), &numeric, compar3); printf("\nsorted numerically:\n"); for(i = 0; i < Sizeofarray(data); i++) printf("%s\n", data[i]); } int compar3(void *userdata, const void *p1, const void *p2) { int numeric = *(int *)userdata; const char *s1 = *(const char **)p1; const char *s2 = *(const char **)p2; if(numeric) return atoi(s1) > atoi(s2); else return strcmp(s1, s2); } 函数,一个用于字母数据,一个用于数字。我可以将一个函数或另一个函数传递给compar,如下所示:

qsort

这样我根本不需要qsort(data, Sizeofarray(data), sizeof(*data), numeric ? compar_num : compar_alph); 指针,或userdata或全局变量。但我希望这个例子已经证明了qsort_r指针如何有用,以及如何使用它。