问题描述
我无法在Vala中编写异步函数。
我知道该函数的回调必须在某个位置进行注册,以便在yield语句之后继续执行,因此我将其添加到GLib主循环中,以使用Idle.add
进行定期调用。这是我的伪代码:
async void loop_func()
{
var id = Idle.add(loop_func.callback);
message("Added idle %u",id);
yield;
message("yielded successfully 1");
yield;
message("yielded successfully 2");
for (var i = 0; i < 20; i++)
{
message("%d",i);
yield;
}
message("finished");
}
int main()
{
var loop = new GLib.MainLoop();
loop_func.begin(() => {loop.quit();});
loop.run();
return 0;
}
尽管如此,第二个yield语句之后的代码从未执行过。 从输出中可以看到:
$ ./async
** Message: 20:07:24.932: async.vala:4: Added idle 1
** Message: 20:07:24.932: async.vala:7: yielded successfully 1
然后挂起。
我在这里想念什么?
解决方法
尽管GLib反复调用一个空闲函数,直到它告诉它停止(通过返回false
),但异步函数的.callback
似乎在调用后立即停止重复。
这可以通过在yield语句之前每次都添加回调来克服:
async void loop_func()
{
for (var i = 0; i < 20; i++)
{
Idle.add(loop_func.callback);
yield;
}
}
执行此操作的另一种方法(可能更有效)是忽略异步函数的回调返回的值:
async void loop_func()
{
var id = Idle.add(() => {loop_func.callback(); return Source.CONTINUE;});
for (var i = 0; i < 20; i++)
{
yield;
}
Source.remove(id);
}