4

我正在使用瓦拉。这是给出编译时错误的源代码:

private Gee.HashMap<string,VoidFunc> fill_actions()
{
    var actions = new Gee.HashMap<string,VoidFunc>();
    MainWindow win = window;
    actions["t"] = () => _puts(win.title);
    return actions;
}

首先,我尝试直接访问 this.window ,但这给出了另一个错误,所以我使用本地范围变量进行了尝试。

直接执行 this.window 时出错:

This access invalid outside of instance methods
4

1 回答 1

9

听起来 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;
  }
}
于 2012-07-03T17:30:55.053 回答