当前位置: 首页 > news >正文

Lua 笔记--编译、执行、错误与协同程序

        一般dofile 可以这样来定义:

function dofile(filename)
    local f = assert(loadfile(filename))
    return f()
end

注意,如果loadfile 失败,那么其中assert 就会引发一个错误。

        函数loadstring 与loadfile 类似,不同之处在于它是从一个字符串中读取代码,而非从文件读取。例如,如下代码:

f = loadstring("i = i + 1")

f 就变成了一个函数,每次调用时就执行“i = i + 1".

i = 32
local i = 0
f = loadstring("i = i + 1"; print(i)")
g = function() i = i + 1; print(i) end
f()        -->33
g()        -->1

        函数g 如预期地操作了局部的i, 但是f 操作的却是全局的i, 这是因为loadstring总是在全局环境中编译它的字符串。loadstring最典型的用处就是执行外部代码,也就是那些位于程序之外的代码。

        assert 函数检查其第一个参数是否为true,若为true,则简单地返回该参数;否则(为false 或nil)就引发一个错误。它的第二个参数是一个可选的信息字符串。注意,assert 是一个正规的函数,所以Lua同样会在调用该函数前对其参数求值。

file  = assert(io.open("te", "r"))
-->stdin:te:No such file or directory

        上例中,如果io.open 失败了,assert就引发了一个错误。

        pcall 函数会以一种”保护模式“来调用它的第一个参数,因此pcall 可以捕获函数执行中的任何错误。如果没有发生错误,pcall会返回true 及函数调用的返回值;否则,返回false 及错误消息。


        协同程序与线程差不多,也就是一条执行序列,拥有自己独立的栈、局部变量和指令指针,同时又与其他协同程序共享全局变量和其他大部分东西。协同程序需要彼此协作地运行,也就是会说,一个具有多个协同程序的程序在任意时刻只能运行一个协同程序,并且正在运行的协同程序只会在其显示的要求挂起时,它的直到才会暂停。

        Lua将所有关于协同程序的函数放置在一个名为”coroutine“的table 中。函数create 用于创建新的协同程序,它只有一个参数,就是一个函数。create 会返回一个thread 类型的值,用以表示新的协同程序。

co = coroutine.create(function() print("hi") end)
print(co)        -->thread:0x8071d98

        一个协同程序可以处于4中不同的状态:挂起(suspended)、运行(running)、死亡(dead)和正常(normal)。当创建一个协同程序时,它处于挂起状态。

print(coroutine.status(co))        -->suspended

        函数coroutine.resume 用于启动或再次启动一个协同程序的执行,并将其状态由挂起改为运行:

coroutine.resume(co)            -->hi

        本例中,协同程序的内容只是简单地打印了”hi“后便终止了,然后它就处于死亡状态,也就再也无法返回了:

print(coroutine.status(co))        -->dead

        而协同程序的真正强大之处在于函数yield 的使用上,该函数可以让一个运行中的协同程序挂起,而之后可以再恢复它的运行:

co = coroutine.create(function()
    for i=1, 10 do
        print("co", i)
        coroutine.yield()
    end
end

        现在当唤醒这个协同程序时,它就会开始执行,直到第一个yield:

coroutine.resume(co)        -->co    1
print(coroutine.status(co))        -->suspended
coroutine.resume(co)         -->co    2
coroutine.resume(co)         -->co    3
...
coroutine.resume(co)         -->10
coroutine.resume(co)         --什么都不打印

         请注意,resume是在保护模式中运行的。因此,如果在一个协同程序的执行中发生任何错误,Lua是不会显示错误消息的,而是将执行权返回给resume调用。

        当一个协同程序A唤醒另一个协同程序B时,协同程序A就处于一个特殊状态,既不是挂起状态也不是运行状态,这时的状态称为”正常“状态。

co = coroutine.create(function(a, b, c)
    print("co", a, b, c)
    end)
coroutine.resume(co, 1, 2, 3)        -->co    1    2    3

        在resume 调用返回的内容中,第一个值为true 则表示没有错误,而后面所有的值都是对应yield 传入的参数:

co = coroutine.create(function(a, b)
    coroutine.yield(a + b, a - b)
    end)
print(coroutine.resume(co, 20, 10)        -->true    30    10

        当一个协同程序结束时,它的主函数所返回的值都将作为对应resume 的返回值:

co = coroutine.create(function()
    return 6, 7
    end)
print(coroutine.resume(co))        -->true    6    7

        一个关于协同程序的经典示例就是”生产者-消费者“的问题:

--生产者
function producer()
while true do
    local x = io.read()        --产生新的值
    send(x)                    --发送给消费者
end
end

--消费者
function consumer()
while true do
    local x = receive()        --从生产者接收值
    io.write(x, "\n")          --消费新的值
end
end

function receive()
    local status, value = coroutine.resume(producer)
    return value
end

function send(x)
    coroutine.yield(x)
end

相关文章:

  • 解决input file按钮要点击两次才弹出选择文件窗口
  • java-NIO
  • 转:对分库分表的一些想法
  • Web 图形验证码的校验1
  • Xcode快捷键--灰常实用的快捷键,以后编程快捷多了
  • 十条nmap常用的扫描命令
  • winPE系统下查看系统的IP及DNS地址
  • PC标签 phpcmsv9
  • YII 验证和消息
  • Postgresql数据库运维笔记
  • C#中箱子、扳手、闪电等含义
  • 转载: #pragma pack(push,1) #pragma pack(pop)
  • 【转】Android 防破解技术简介
  • Go 环境变量
  • 使用LocalBroadcastManager
  • 0基础学习移动端适配
  • classpath对获取配置文件的影响
  • create-react-app做的留言板
  • Hexo+码云+git快速搭建免费的静态Blog
  • Laravel深入学习6 - 应用体系结构:解耦事件处理器
  • MD5加密原理解析及OC版原理实现
  • Python 反序列化安全问题(二)
  • 阿里云应用高可用服务公测发布
  • 代理模式
  • 好的网址,关于.net 4.0 ,vs 2010
  • 欢迎参加第二届中国游戏开发者大会
  • 回顾 Swift 多平台移植进度 #2
  • 验证码识别技术——15分钟带你突破各种复杂不定长验证码
  • 新海诚画集[秒速5センチメートル:樱花抄·春]
  • ​ 无限可能性的探索:Amazon Lightsail轻量应用服务器引领数字化时代创新发展
  • ​ubuntu下安装kvm虚拟机
  • # Python csv、xlsx、json、二进制(MP3) 文件读写基本使用
  • (16)UiBot:智能化软件机器人(以头歌抓取课程数据为例)
  • (Spark3.2.0)Spark SQL 初探: 使用大数据分析2000万KF数据
  • (附源码)apringboot计算机专业大学生就业指南 毕业设计061355
  • (十六)Flask之蓝图
  • (总结)Linux下的暴力密码在线破解工具Hydra详解
  • .NET 8.0 发布到 IIS
  • .Net 应用中使用dot trace进行性能诊断
  • .NET/C# 利用 Walterlv.WeakEvents 高性能地定义和使用弱事件
  • .net程序集学习心得
  • .net下简单快捷的数值高低位切换
  • .NET学习教程二——.net基础定义+VS常用设置
  • [51nod1610]路径计数
  • [ABP实战开源项目]---ABP实时服务-通知系统.发布模式
  • [AutoSAR系列] 1.3 AutoSar 架构
  • [BIZ] - 1.金融交易系统特点
  • [bzoj1901]: Zju2112 Dynamic Rankings
  • [c]统计数字
  • [C++] 默认构造函数、参数化构造函数、拷贝构造函数、移动构造函数及其使用案例
  • [C++参考]拷贝构造函数的参数必须是引用类型
  • [C++打怪升级]--学习总目录
  • [DevEpxress]GridControl 显示Gif动画
  • [Foreman]解决Unable to find internal system admin account
  • [GPT]Andrej Karpathy微软Build大会GPT演讲(上)--GPT如何训练