Lua协程介绍

Lua协程

协程

什么是协程

协程(coroutine)是一种程序组件,它允许执行代码在不同的点暂停和恢复,从而实现协作式的多任务处理。与传统的线程不同,协程不是由操作系统调度的,而是由程序自身控制的。因此,协程提供了一种轻量级的并发机制,适用于需要多任务但不需要真正并行的场景

协程的特点

1.轻量级:协程不需要占用如线程堆栈等额外的系统资源,因此创建和切换协程的开销非常低。

2.非抢占式:协程的切换是显式的,即只有当协程主动让出控制权时,程序才会切换到另一个协程。

3.合作式多任务:多个协程之间通过显式的让出和恢复控制权来进行合作,从而实现多任务处理。

4.独立的执行上下文:每个协程都有自己的执行上下文,包括局部变量等。

协程和线程的区别

1.调度方式:线程由操作系统调度,而协程由程序自身调度。

2.并行与并发:线程可以真正并行执行(多核 CPU),而协程在任何时刻只能有一个在运行(单线程)。

3.创建和切换开销:线程的创建和上下文切换开销较大,而协程的创建和切换开销较小。

协程的工作原理

协程的核心机制是允许代码在某个点暂停(挂起),并在需要时从暂停的地方继续执行。这通常通过以下两种操作来实现:

挂起(yield):暂停协程的执行,并返回控制权给调度者(通常是调用协程的函数)。

恢复(resume):从挂起点继续执行协程。

Lua协程示例

创建协程:使用 coroutine.create 创建一个新的协程。

启动协程:使用 coroutine.resume 启动或恢复一个协程。

挂起协程:使用 coroutine.yield 挂起当前协程,并将控制权交回给调用者。

状态查询:使用 coroutine.status 查询协程的状态,状态可以是 suspended(挂起)、running(运行中)、normal(正常)或 dead(结束)。

主协程:每个 Lua 程序都有一个主协程,可以通过 coroutine.running 获取当前正在运行的协程。

lua 复制代码
-- 模拟一个简易的异步任务示例
function longRunningTask()
    local value = 0
    for i = 1, 3 do
        value = value + 1
        print("Running step", i, "with value", value)
        coroutine.yield(value)
    end
    return value
end

-- 创建协程
local co = coroutine.create(longRunningTask)

--[[
coroutine.resume 返回两个值:

布尔值,表示协程是否成功执行。true 表示协程成功执行,false 表示遇到了错误。
其他返回值,它们依赖于协程内部的执行情况:
如果协程在 coroutine.yield 处暂停,该处的返回值会作为 resume 的第二个返回值。
如果协程正常结束,coroutine.resume 的第二个返回值是协程结束时coroutine.yield(*)传入的返回值。
如果协程遇到运行错误,coroutine.resume 的第二个返回值是错误消息。

--]]

-- 恢复协程,接收每次 yield 返回的值
local status, result = coroutine.resume(co)  -- 输出 "Running step 1 with value 1"
local r = coroutine.status(co)
print("Received from yield:", status,result,r)
-- can todo something... 处理其他任务,等待下一次启动协程
status, result = coroutine.resume(co)  -- 输出 "Running step 2 with value 2"
r = coroutine.status(co)
print("Received from yield:", status,result,r)
-- can todo something...
status, result = coroutine.resume(co)  -- 输出 "Running step 3 with value 3"
r = coroutine.status(co)
print("Received from yield:", status,result,r)
-- can todo something...
status, result = coroutine.resume(co)  -- 协程完成,返回最终结果
r = coroutine.status(co)
print("Final result:", status,result,r)
相关推荐
majingming1233 小时前
FUNCTION
java·前端·javascript
zopple3 小时前
常见的 Spring 项目目录结构
java·后端·spring
xuxie995 小时前
N11 ARM-irq
java·开发语言
cjy0001115 小时前
springboot的 nacos 配置获取不到导致启动失败及日志不输出问题
java·spring boot·后端
wefly20175 小时前
从使用到原理,深度解析m3u8live.cn—— 基于 HLS.js 的 M3U8 在线播放器实现
java·开发语言·前端·javascript·ecmascript·php·m3u8
zhenxin01226 小时前
Spring Boot实现定时任务
java
小江的记录本6 小时前
【事务】Spring Framework核心——事务管理:ACID特性、隔离级别、传播行为、@Transactional底层原理、失效场景
java·数据库·分布式·后端·sql·spring·面试
sheji34166 小时前
【开题答辩全过程】以 基于springboot的校园失物招领系统为例,包含答辩的问题和答案
java·spring boot·后端
寂静or沉默6 小时前
2026最新Java岗位从P5-P7的成长面试进阶资源分享!
java·开发语言·面试
卓怡学长6 小时前
m289在线交友系统
java·spring·tomcat·maven·intellij-idea·hibernate