Lua中编译执行代码相关的函数以及机制
2015-04-09 19:15
609 查看
可以说Lua之所以称为是一种解释型的语言,正是因为有诸如load这样的函数,因为这样的函数使得Lua可以执行动态生成的代码。下***体来分析这些函数。
load函数
load函数原型如下:
如果没有语法错误,则load返回chunk对于的函数,否则返回nil和对应的错误信息。如果返回的函数有upvalue,则第一个upvalue的值被设置成传入的参数env,若没有传入这个参数的值,则第一个upvalue的值为全局环境。返回的函数upvalue不会任何其他函数共享。
参数chunkname作为错误信息的chunk的名字,用于调试。若chunk是字符串,则chunkname默认值是变量chunk的值,否则是=(load)。参数mode决定了传入给chunk的值是文本或二进制数据。如果是”b”,则传给chunk的值必须是二进制数据(比如可以使用string.dump(foo)得到相应函数的二进制数据,然后传递给load),如果是”t”,而必须是文本数据,如果是”bt”,则文件或二进制数据都可以。默认值是”bt”。
Lua不会检测二进制数据的稳定性,因此恶意的二进制数据可能导致Lua解释器Crash。
loadfile函数
函数原型如下:
dofile函数
函数原型如下:
上面讨论的Lua层面的编译执行相关的函数,对应的C API相关函数有:lua_load、luaL_loadfilex、luaL_loadfile和luaL_dofile,这些函数非常类似对应的Lua层函数。
loadstring函数
函数原型如下:
注意该函数在Lua5.2被删除了,因为它完全可被load替代。
参考资料
《Lua程序设计》(第二版)
http://www.lua.org/manual/5.2/manual.html
load函数
load函数原型如下:
load (chunk [, chunkname [, mode [, env]]])该函数加载一个chunk,如果没有错误,则返回一个函数。如果传入chunk的值是一个字符串,则就加载这个字符串;如果传入chunk的值是一个函数,则这个函数必须返回一个字符串,并且load会一直调用这个函数,直到这个函数返回空字符串或nil,load会把每次函数的返回结果连接起来作为chunk去加载。
如果没有语法错误,则load返回chunk对于的函数,否则返回nil和对应的错误信息。如果返回的函数有upvalue,则第一个upvalue的值被设置成传入的参数env,若没有传入这个参数的值,则第一个upvalue的值为全局环境。返回的函数upvalue不会任何其他函数共享。
参数chunkname作为错误信息的chunk的名字,用于调试。若chunk是字符串,则chunkname默认值是变量chunk的值,否则是=(load)。参数mode决定了传入给chunk的值是文本或二进制数据。如果是”b”,则传给chunk的值必须是二进制数据(比如可以使用string.dump(foo)得到相应函数的二进制数据,然后传递给load),如果是”t”,而必须是文本数据,如果是”bt”,则文件或二进制数据都可以。默认值是”bt”。
Lua不会检测二进制数据的稳定性,因此恶意的二进制数据可能导致Lua解释器Crash。
loadfile函数
函数原型如下:
loadfile ([filename [, mode [, env]]])该函数类似于load,但是从文件filename或标准输入(若filename为空)中活动chunk加载。
dofile函数
函数原型如下:
dofile ([filename])读取文件filename的内容,并把它作为lua的chunk执行。如果没传入参数,则从标准输入中读取执行内容。返回值跟chunk执行后返回的值相同。如果出现错误,dofile会把错误信息传递给它的调用者。也就是说,dofile在无保护模式下执行。注意该函数不同于loadfile,loadfile只是返回一个函数,并不会执行,并且出现错误时,loadfile会返回一个错误信息,而不会传递给它的调用者。总之、loadfile会从一个文件中加载Lua代码块,但不会运行代码,只是编译代码,然后将编译结果从一个函数返回。而dofile除了编译代码外,还会运行返回的结果。
上面讨论的Lua层面的编译执行相关的函数,对应的C API相关函数有:lua_load、luaL_loadfilex、luaL_loadfile和luaL_dofile,这些函数非常类似对应的Lua层函数。
loadstring函数
函数原型如下:
loadstring (string [, chunkname])该函数通用类似于load,它等价于给load函数传递字符串做作为参数。为了加载和运行一个字符串,可以这些写:
assert(loadstring(s))()
注意该函数在Lua5.2被删除了,因为它完全可被load替代。
参考资料
《Lua程序设计》(第二版)
http://www.lua.org/manual/5.2/manual.html
相关文章推荐
- Lua中编译执行代码相关的函数详解
- 如何设置VS2008 辅助编译代码提示相关函数以及相关函数的参数
- php代码执行文件重命名函数的权限以及参数
- HM编码器代码阅读(7)——整个编码流程以及相关的函数
- 关于JavaScript预编译和执行顺序以及函数引用类型的思考
- JVM原理(Java代码编译和执行的整个过程+JVM内存管理及垃圾回收机制)
- Fragment的生命周期以及相关库函数的执行流程总结
- luajit笔记---编译成静态库以及FFI绑定宿主程序函数
- luajit笔记---编译成静态库以及FFI绑定宿主程序函数
- SQL Server 第四堂课,创建存储过程。存储过程是一组编译在单个执行计划中的transact-SQL语句。存储过程相当于C#函数,可以允许模块化程序设计,允许更快执行如果某操作需要大量transct-SQL代码或需要重复执行,将在创建存储过程中对其进行分析和优化。
- main 函数执行以前以及以后,分别还会执行什么代码?
- js代码执行过程,js预编译,变量声明提升,函数体整体提升
- c#获取当前代码运行的文件名、运行的函数名以及当前代码执行的行号
- linux的ubuntu上如何编译C和C++代码写的动态库,以及调用执行
- IL代码底层运行机制之函数相关
- main 函数执行以前以及以后,分别还会执行什么代码?(转载)
- Arm汇编学习笔记(二)——编写编译并执行依赖外部模块的汇编代码以及PIC代码分析
- JVM原理(Java代码编译和执行的整个过程+JVM内存管理及垃圾回收机制)
- C#代码的编译与执行过程(摘自ms inside c#)
- Hook API相关技术以及例子,Hook API的原理其实是通过核心函数强制修改原API的头部指针