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)
相关推荐
职略2 小时前
负载均衡类型和算法解析
java·运维·分布式·算法·负载均衡
A22742 小时前
LeetCode 196, 73, 105
java·算法·leetcode
容若只如初见3 小时前
项目实战--Spring Boot + Minio文件切片上传下载
java·spring boot·后端
阿里巴巴P8资深技术专家3 小时前
Java常用算法&集合扩容机制分析
java·数据结构·算法
weixin_440401693 小时前
分布式锁——基于Redis分布式锁
java·数据库·spring boot·redis·分布式
码农爱java3 小时前
Spring Boot 中的监视器是什么?有什么作用?
java·spring boot·后端·面试·monitor·监视器
zengson_g3 小时前
当需要对大量数据进行排序操作时,怎样优化内存使用和性能?
java·数据库·算法·排序算法
血战灬狂龙4 小时前
pom.xml文件加载后没有变成maven图标
xml·java·maven
无名指的等待7124 小时前
SpringBoot实现图片添加水印(完整)
java·spring boot·后端
胡尚4 小时前
Ratf协议图解、Nacos CP集群源码分析
java·spring boot