如果您拥有的只是指向回调函数的函数指针,则无法将额外的数据传递到回调函数中。
例如,假设我有一些字符串想要使用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/Linuxsort
命令及其-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
将它们“转换”为 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]);
}
这也有效。但当然没有人喜欢全局变量。
这就是“userdata”参数的概念出现的地方。我不知道它有多标准,但我的系统提供了一个qsort
变体称为qsort_r
。 (这 ”r
”代表“可重入”。)在这个版本中,比较函数传递了一个额外的参数,我可以用它做任何我想做的事情。在这里,我可以将它作为一个指向我的numeric
旗帜,现在是我的numeric
flag 不必是全局变量:
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
,或者是一个全局变量。但我希望这个例子能够展示如何userdata
指针很有用,以及如何使用它。