使用函数时的一些要点

(与主题无关)

数字型for 的格式

exp3默认为1

for var=exp1 , exp2 , exp3  do
<执行体>
end

泛型for迭代的类型

标准库中提供了几种迭代器

io.lines() 用于迭代文件中每行
pairs() 迭代table中元素
ipairs() 迭代 数组 元素
string.gmatch() 迭代字符串中单词

函数

多重返回值: x , y = foo()

foo()返回值就是2个值,并不是table或者其他类型。
t = { foo() } 这是再封装 成为一个table。{ “a” , “b” }

变长参数: function add ( ... )

在函数内部使用这些参数 依然使用 ... 来表示,此时,这个表达式 是一个像多重返回值一样的序列,不是table。当然,可以再包装 { ... },这也是通常函数内遍历变长参数的方式。

function fwrite ( fmt , ... )
return io.write( string.format( fmt, ...) )
end

具名实参

w = Window{ x=0 , y=0 , width=300 , height=200,
        title = “Lua” , background=”blue”,
        border = true }

Window函数只有一个参数(当函数只有一个参数时,调用时可以去掉圆括号),并且,这个参数需要我们用table,就可以实现具名参数了。

深入函数

第一类值:函数(这个函数本身,而非其返回值,其实现我们可以理解为类似于C中的函数指针被存储)可以存储到变量中,或者table中,可以作为实参传递给其他函数,还可以作为作为其他函数的返回值。词法域:类似于JS中的闭包。
函数名,一个拥有某函数的变量。函数与所有其他值一样都是匿名的。

foo = function( x )  return 2*x  end

table.sort ( onetable , function( a , b ) 
                            return (a.name > b.name) 
                        end 
)

对一些函数的访问,想要建立一个安全的运行环境,比如,限制一个程序访问文件:

do
    local  oldOpen = io.open
    local 	access_OK = function ( filename , mode )
                        <检查访问权限>
    end

    io.open  =  function ( filename , mode )
        if 	access_OK( filename , mode ) 	then
            return 	 oldOpen( filename , mode )
        else
            return  nil , “access denied”
        end
    end
end

正确的尾调用

function f ( x )  return g( x )  end  ---yes
function f ( x )   g( x )  end  ---not

“尾调用”不需要消耗任何栈信息,即,当g返回时,程序执行点将直接返回 调用f的地方。可以理解为,在f返回并调用g的时候,f 已经被清栈,之后接着调用g,g结束后自然栈调用信息直接到了,f的调用者的地方。
“尾调用” return [func] ( [args] )
非常适合状态机一类程序。

截取一段例子:

funciton room1 ()
    local move = io.read()
    if  move == “south” then return room3()
    elseif  move == “east” then return room2()
    else
        print( “invalid move” )
        return room1()  --stay in the same room
    end
end