听起来 VoidFunc 是用 [CCode (has_target = false)] 声明的。这意味着没有上下文信息传递给它,而 AFAIK 是委托作为泛型类型参数工作的唯一方式。这样做的原因是 C 语言的限制,所以假设 VoidFunc 看起来像这样:
[CCode (has_target = false)]
public delegate void VoidFunc ();
你会在 C 中得到的东西是这样的:
typedef void (*VoidFunc)();
如果您没有 [CCode (has_target = false)],则与类似的情况相反:
typedef void (*VoidFunc)(gpointer user_data);
当您在 C 中传递回调时,通常会使用一到三个参数。所有这三个的东西看起来像这样:
void foo (VoidFunc void_func, gpointer user_data, GDestroyNotify notify);
第一个参数是实际功能。第二个参数是作为 user_data 传递给回调的值,它是 Vala 用来将上下文信息传递给回调的值(这允许它充当实例方法,甚至是闭包)。第三个参数用于指定在不再需要时释放 user_data 的函数。
[CCode (has_target = false)] 的意思是委托没有 user_data 参数,因此不能用作闭包或实例方法。
这对于泛型参数来说是必要的原因是泛型在 C 级别看起来像这样:
void foo_bar (gpointer data, GDestroyNotify notify);
第一个参数是您要用作通用值的数据,第二个参数实际上只有在拥有通用参数时才添加(就像在 Gee 中的 set 方法的情况下一样),并使用 user_data 作为不再需要 user_data 时的参数。
如您所见,当尝试将委托用作泛型时,没有地方可以放置 user_data 参数,这就是为什么 Vala 只允许没有目标的委托作为泛型参数。
解决方案基本上是将委托包装在一个类中:
public delegate void VoidFunc ();
public class YourClass {
private class VoidFuncData {
public VoidFunc func;
public VoidFuncData (owned VoidFunc func) {
this.func = (owned) func;
}
}
private Gee.HashMap<string,VoidFuncData> fill_actions() {
var actions = new Gee.HashMap<string,VoidFuncData>();
string win = "win";
actions["t"] = new VoidFuncData (() => GLib.debug (win));
return actions;
}
}