chapter8_1 编译执行和错误

 

1、编译

  前面介绍的,dofile是一个运行lua代码块的基本操作,实际上它是一个辅助函数。

loadfile才真正做了核心的工作。dofile(打开文件,执行里面的代码块)和loadfile(从文件或stdin中获取代码块)实际不运行代码,

只是编译,然后将结果作为一个函数返回。loadfile与dofile不同的 是它不会引发错误,只是返回错误值并不处理错误。

一般dofile定义:

function dofile(filename)
    local f = assert(loadfile(filename))   --如果loadfile失败,assert就会引发一条错误
    return f()
end

通常,如果我们要多次运行一个文件,只需用loadfile一次,再多次调用它的返回结果就可以了。

相对于dofile来说,它只编译一次,开销就小得多。

与loadfile类似的函数load,它是从字符串中读取代码,而不是文件。

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

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

i = 0
f();print(i)    --> 1
f();print(i)    --> 2

使用load的时候要小心,它的开销大,并且会返回一些难以理解的代码。

如果你想塑造一个便捷但粗糙的dostring(load and run a chunk):

load(s)()   -- 有语法错误,load返回nil就会报错

--优化的版本

assert(load(s))()

一般将load用于字面字符串是没有意义的:

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

--基本上等价于

f = function() i = i + 1 end

但是第二行要快的多,因为它只编译了一次,而第一行每次调用load都要重新编译。

因为load编译的时候不涉及词法域,所以两者还是有区别的:

i  =  32
local i = 0
g = load("i = i + 1 ; print(i)")        -- 字符串,访问的是全局  i
f = function() i = i + 1 ;print(i) end  -- 函数块,访问的是local  i
g()          --> 33
f()          --> 1

函数 f 访问的是local  i

函数 g 访问的是全局  i

因为load总是在全局环境去编译代码块。

load最典型的用处是执行外部代码,那些位于程序之外的代码( 就像加载配置文件config一样,之前疑惑的地方 )。

有很多地方需要用户自定义函数,就要求输入函数代码,然后调用load去求值。

如果要对一个表达式求值,则必须在其之前添加return,这样才能构成一条语句,返回表达式的值:

print"enter your expression:"        -- 输入 1 + 2
local r = io.read()
local func = assert(load("return " .. r))
print("the value is " .. func())       -- the value is 3

因为load返回的是函数,所有可以多次调用:

print "enter function to be plotted(with variable 'x'):"  --输入一个数字,因为string.rep(第二个参数要数字,如果输入a或者b之类的 返回值为nil)
local r = io.read()
local f = assert(load("return " .. r))
for i = 1,20 do 
    x = i      
    print(string.rep("*",f()))
end

loadfile的实质也是调用load,load接收一个"读取器函数--reader function",并在内部调用它来获取程序块:

它可以分几次返回一个程序块,load会反复调用它,直到返回nil。

下面的例子就等于loadfile:

f = load(io.lines(filename,"*L")) --每次调用io.lines,都返回一个函数

--下面的代码更高效

f = load(io.lines(filename,1024))

Lua把这些独立的代码块当作匿名函数的函数体。比如load(“a = 1”)等于:

function (...) a = 1 end

代码块也可以声明local变量:

f = load("local a = 10; print(a+20)")
f()   --> 30

有了这些特点:可以重写上面的例子,避免用全局变量:

print "enter function to be plotted(with variable 'x'):"
local r = io.read()
local f = assert(load("local x = ...;return " .. r)) --把x声明为local变量,当调用f(i)时,...就变成了参数i。
for i = 1,20 do
    print(string.rep("*",f(i)))
end

load不会引发错误,在错误情况中,load会返回nil及一条错误消息。

print (load("i i"))
--> nil   [string "i i"]:1: syntax error near 'i'

此外,这些函数不会有其他副作用。

有一个误解,认为加载了一块代码,就定义了其中的函数。

在Lua中函数定义是一种赋值操作,也就是说,它们是在运行时才完成的操作:

一个foo.lua:

--file 'foo.lua'
function foo(x)
    print(x)
end

运行它:

f = loadfile("foo.lua")

运行之后,foo函数只是编译,还没有被定义。为了定义它,必须运行这个chunk:

print(foo)        --> nil
f()               -- defines 'foo',即运行它
foo("ok")         --> ok

此外,若需要在一个商业品质的程序中执行外部代码,还要处理加载程序块时报告的错误。

如果代码是不可信的,还要在一个保护环境下执行。

 

posted @ 2016-08-27 16:49  daiker  阅读(188)  评论(0编辑  收藏  举报