- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试提出一种解决方案,用于从 C 函数生成 Luajit 协程,该函数立即创建一个 tasklet 以在另一个操作系统线程上进行处理。
根据各种 Lua 文档,事情开始相互严重矛盾,这不是完全可能的吗?文档不是很清楚,也没有解释原因。
Lua 5.1 规定每个协程都有一个堆栈。但是,只有一个全局 C 堆栈。我不完全确定为什么这是一个障碍。
Lua 5.2 显然用 lua_pcallk 和 lua_yieldk 修复了这个问题。但是解释非常困惑。
但这些都没有说明我正在使用的 VM...即 LuaJIT 2.0.4 和 LuaJIT 2.1.0。
谷歌搜索告诉我 Luajit 1.x 实现了 CoCo,它显然为每个 lua 线程(协程)使用了真正的 C 堆栈。这允许从任何地方屈服。
只有一次搜索让我看到 LuaJIT 2.x 显然没有实现 coco,因为每个协程都使用 C 堆栈。
谁能告诉我从 C 生成协程的问题是什么?并验证我是否可以从 c 安全地产生/恢复 luajit 2.x 协程?
最佳答案
在引用Lua实现中,每个Lua协程都有自己的Lua栈,它只是lua_State
中的一个数组,与C栈无关。 Lua 无法保存 C 堆栈(因为这在标准 C 中是不可能的),因此如果当前正在执行 C 函数,则它无法生成协程。
例如,如果您有 Lua 函数 a
调用 C 函数 b
调用 Lua 函数 c
,并且 c
尝试让步,Lua 将无法保存 b
的局部变量(因为它是 C 函数)并且会失败。
这也适用于很多内置的 Lua 函数。正如您提到的,在 Lua 5.1 中,实现不支持在 pcall
中产生,直到 Lua 5.2 显然添加了特殊函数以使其工作。
Coco 是标准 Lua 实现的补丁,它在协程中实现了单独的 C 堆栈,因此 Lua 现在可以“保存”C 函数变量。显然 LuaJit 1.x 也包含它。它不适用于 LuaJit 2.x,因为它是 Lua 的完全不同的实现。
LuaJit 2.x 在 Extensions page 中有以下段落:
Fully Resumable VM
The LuaJIT VM is fully resumable. This means you can yield from a coroutine even across contexts, where this would not possible with the standard Lua 5.1 VM: e.g. you can yield across pcall() and xpcall(), across iterators and across metamethods.
显然,跨内置函数的 yield 应该可以工作,尽管它是否适用于任意 Lua C API 函数仍然是模棱两可的。然而,它很容易测试;编写一个简单的 C API 函数,它接受一个 Lua 函数并调用它,然后将一个函数传递给它,该函数会产生。如果它不起作用,它应该抛出一个错误。
请注意,使用 FFI 加载的普通 C 函数根本不允许触及 Lua 状态。。这包括尝试屈服。
关于你能在 C 的任何地方 Yield 和 Resume Luajit 协程吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40682722/
function* generatorFunction() { yield (yield 1)(yield 2)(yield 3)(); } var iterator = generatorFun
ECMAScript 6 应该带来生成器函数和迭代器。生成器函数(具有 function* 语法)返回一个迭代器。迭代器有一个 next 方法,当重复调用时,该方法会执行生成器函数的主体,并在每个 y
ECMAScript 6 应该引入生成器函数和迭代器。生成器函数(具有 function* 语法)返回迭代器。迭代器有一个 next 方法,当重复调用时,它会执行生成器函数的主体,在每个 yield
关闭。这个问题需要details or clarity .它目前不接受答案。 想改进这个问题吗? 通过 editing this post 添加细节并澄清问题. 关闭 2 年前。 Improve t
自 python 2.5 以来,可以将 send()、throw()、close() 放入生成器中。在定义的生成器中,可以通过执行以下操作来“捕获”发送的数据: def gen(): whil
return的区别和 yield似乎很清楚,直到我发现还有 yield from以及将两者结合起来的可能性 return和 yield在完全相同的功能! 我对return的理解之后的一切都是 不是 执
假设我有这个部分,我正在尝试渲染 #layouts/_subheader.html.erb 当我在这样的 View 中使用这个部分时 Content For Yield
yield操作符是由编译器在底层实现的,该编译器生成一个实现符合 IEnumerable 的状态机的类。和 IEnumerator . 给定一个罗斯林 MethodDeclarationSyntax
$item) echo "$index $item" . PHP_EOL; } resolve(generator1()); echo PHP_EOL; resolve(gener
这个问题在这里已经有了答案: Why converting list to set is faster than converting generator to set? (1 个回答) List c
是否有一个单行代码来获取生成器并生成该生成器中的所有元素?例如: def Yearly(year): yield YEARLY_HEADER for month in range(1, 13)
刚发现yield from 结构,在我看来这有点像反向的yield,而不是从生成器中获取对象,您插入/将对象发送到生成器。喜欢: def foo(): while True:
考虑以下代码: def mygen(): yield (yield 1) a = mygen() print(next(a)) print(next(a)) 输出产量: 1 None 解释器
Guido van Rossum,在 2014 年关于 Tulip/Asyncio 的演讲中 shows the slide : Tasks vs coroutines Compare: res =
谁能帮我理解“yield self”和“yield”的区别? class YieldFirstLast attr_accessor :first, :last def initiali
这是我目前使用 Laravel 5 实现的 Open Graph 标签: app.blade.php @yield('title') page.blade.php @extends('app'
在 Tornado 中,我们通常会编写如下代码来异步调用函数: class MainHandler(tornado.web.RequestHandler): @tornado.gen.coro
本文整理了Java中aQute.bnd.indexer.analyzers.Yield.yield()方法的一些代码示例,展示了Yield.yield()的具体用法。这些代码示例主要来源于Github
我们有超过 100 个共同基金的每日返回,我们希望将这些返回转换为月度返回。每月返回不应是每个月的平均值,而是每个月末的资金返回。基金在不同的时间点开始和结束,它们需要自己保留(不是每个月的共同基金
如何实现 C# yield return使用 Scala 延续?我希望能够编写 Scala Iterator s 风格相同。在 this Scala news post 的评论中有刺伤,但它不起作用(
我是一名优秀的程序员,十分优秀!