麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 開發 > 綜合 > 正文

Lua中的函數(function)、可變參數、局部函數、尾遞歸優化等實例講解

2024-07-21 23:04:07
字體:
來源:轉載
供稿:網友

一、函數

在Lua中,函數是作為"第一類值"(First-Class Value),這表示函數可以存儲在變量中,可以通過參數傳遞給其他函數,或者作為函數的返回值(類比C/C++中的函數指針),這種特性使Lua具有極大的靈活性。
 
Lua對函數式編程提供了良好的支持,可以支持嵌套函數。
 
另外,Lua既可以調用Lua編寫的函數,還可以調用C語言編寫的函數(Lua所有的標準庫都是C語言寫的)。
 
定義一個函數

復制代碼 代碼如下:

function hello()
print('hello')
end

 

hello函數不接收參數,調用:hello(),雖然hello不接收參數,但是還可以可以傳入參數:hello(32)
 
另外如果只傳遞一個參數可以簡化成functionname arg的調用形式(注意數值不行)

復制代碼 代碼如下:

> hello '3'
hello
> hello {}
hello
> hello 3
stdin:1: syntax error near '3'

 
另外對變量名也不適用
復制代碼 代碼如下:

> a = 21
> print a
stdin:1: syntax error near 'a'

 
另外,Lua函數不支持參數默認值,可以使用or非常方便的解決(類似Javascript)
復制代碼 代碼如下:

> function f(n)
>> n = n or 0
>> print(n)
>> end
> f()
0
> f(1)
1

 

Lua支持返回多個值,形式上非常類似Python:

復制代碼 代碼如下:

> function f()
>> return 1,2,3
>> end
> a,b,c = f()
> print(a .. b .. c)
123

 
函數調用的返回值可以用于table:
復制代碼 代碼如下:

> t = {f()}
> print(t[1], t[2], t[3])
1        2        3
 
可見,f()返回的三個值分別稱為table的3個元素,但是情況并不總是如此:
復制代碼 代碼如下:

> t = {f(), 4}
> print(t[1], t[2], t[3])
1        4        nil

這次,f()返回的1,2,3只有1稱為table的元素;
復制代碼 代碼如下:

> t = {f(), f()}
> print(t[1], t[2], t[3], t[4], t[5])
1        1        2        3        nil

 
總之:只有最后一項會完整的使用所有返回值(假如是函數調用)。
 
對于無返回值的函數,可以使用(f())的形式強行返回一個值(nil)
復制代碼 代碼如下:

> function g()
>> end
> print(g())
 
> print((g()))
nil

實際上,(f())形式的調用返回一個且只返回一個值
復制代碼 代碼如下:

> print((f()))
1
> print(f())
1        2        3

 

二、變長參數

Lua支持編程參數,使用簡單(借助于table、多重賦值)

復制代碼 代碼如下:

> function f(...)
for k,v in ipairs({...}) do
print(k,v)
end
end
> f(2,3,3)
1        2
2        3
3        3

使用多重賦值的方式
復制代碼 代碼如下:

> function sum3(...)
>> a,b,c = ...
>> a = a or 0
>> b = b or 0
>> c = c or 0
>> return a + b +c
>> end
> =sum3(1,2,3,4)
6
> return sum3(1,2)
3

通常在遍歷變長參數的時候只需要使用{…},然而變長參數可能會包含一些nil;那么就可以用select函數來訪問變長參數了:select('#', …)或者 select(n, …)

 

select('#', …)返回可變參數的長度,select(n,…)用于訪問n到select('#',…)的參數

復制代碼 代碼如下:

> =select('#', 1,2,3)
3
> return select('#', 1,2, nil,3)
4
> =select(3, 1,2, nil,3)
nil        3
> =select(2, 1,2, nil,3)
2        nil        3

注意:Lua5.0中沒有提供…表達式,而是通過一個隱含的局部table變量arg來接收所有的變長參數,arg.n表示參數的個數;

 

三、函數式編程

函數做一個First-Class Value可以賦值給變量,用后者進行調用

復制代碼 代碼如下:

> a = function() print 'hello' end
> a()
hello
> b = a
> b()
hello

匿名函數
復制代碼 代碼如下:

> g = function() return function() print 'hello' end end
> g()()
hello

函數g返回一個匿名函數;
 
閉包是函數式編程的一種重要特性,Lua也支持
復制代碼 代碼如下:

> g = function(a) return function() print('hello'.. a); a = a + 1 end end
> f = g(3)
> f()
hello3
> f()
hello4

 

四、局部函數

局部函數可以理解為在當前作用域有效的函數,可以用local變量來引用一個函數:

復制代碼 代碼如下:

> do
>> local lf = function() print 'hello' end
>> lf()
>> end
hello
> lf()
stdin:1: attempt to call global 'lf' (a nil value)
stack traceback:
stdin:1: in main chunk
[C]: in ?

 

需要注意的是,對于遞歸函數的處理

復制代碼 代碼如下:

> do
local lf = function(n)
if n <= 0 then
return
end
print 'hello'
n = n -1
lf(n)
end
lf(3)
end
hello
stdin:8: attempt to call global 'lf' (a nil value)
stack traceback:
stdin:8: in function 'lf'
stdin:9: in main chunk
[C]: in ?

而應該首先聲明local lf, 在進行賦值
復制代碼 代碼如下:

do
local lf;
lf = function(n)
if n <= 0 then
return
end
print 'hello'
n = n -1
lf(n)
end
lf(3)
end
hello
hello
hello

Lua支持一種local function(…) … end的定義形式:
復制代碼 代碼如下:

> do
local function lf(n)
if n <= 0 then
return
end
print 'hello'
n = n -1
lf(n)
end
lf(3)
end
hello
hello
hello
> lf(3)
stdin:1: attempt to call global 'lf' (a nil value)
stack traceback:
stdin:1: in main chunk
[C]: in ?

 

五、尾調用

所謂尾調用,就是一個函數返回另一個函數的返回值:

復制代碼 代碼如下:

function f()

return g()
end
 
因為調用g()后,f()中不再執行任何代碼,所以不需要保留f()的調用桟信息;Lua做了這樣的優化,稱為"尾調用消除",g()返回后,控制點直接返回到調用f()的地方。
 
這種優化對尾遞歸非常有益,通常遞歸意味著調用桟的不斷增長,甚至可能造成堆棧溢出;而尾遞歸提供了優化條件,編譯器可以優化掉調用桟。
 
下面的遞歸函數沒有使用尾遞歸,而參數為大數時,堆棧溢出:
復制代碼 代碼如下:

> function f(n)
>> if n <= 0 then
>> return 0
>> end
>> a = f(n-1)
>> return n * a
>> end
> f(10000000000)
stdin:5: stack overflow
stack traceback:
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
...
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:5: in function 'f'
stdin:1: in main chunk
[C]: in ?

優化為尾遞歸
復制代碼 代碼如下:

function f(n, now)
if n <= 0 then
return now
end
 
return f(n-1, now*n)
end
f(10000000000, 1)

運行n久也無堆棧溢出;
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 中文字幕视频在线播放 | 欧美一级精品片在线看 | 国产精品亚洲yourport | 91精品国产综合久久男男 | v片在线看 | 男女羞羞视频 | 久久久av亚洲男天堂 | www.99av | 欧美三级美国一级 | 全免费午夜一级毛片真人 | 中国老女人一级毛片视频 | 久久久精品视频网站 | 免费黄色a | 色婷婷tv | 日本aaa一级片 | 国产精品自在线拍 | 看免费一级毛片 | 男女无套免费视频 | 国产精品美女久久久久久不卡 | 小雪奶水翁胀公吸小说最新章节 | 日本精品婷婷久久爽一下 | 精品国产一区二区三区久久久蜜月 | 国产精品久久久久久久模特 | h视频在线播放 | 深夜免费视频 | 久草干| 黑人日比 | 亚洲福利视 | 欧美日韩1区2区3区 黄片毛片一级 | 91精品久久久久久久久 | 成人国产精品久久 | 久久96国产精品久久秘臀 | 欧美人与性禽动交精品 | 国产午夜三级一区二区三桃花影视 | 线观看免费完整aaa 久久不雅视频 | 午夜精品老牛av一区二区三区 | 午夜视频在线免费观看 | 精品国产一区二区三区成人影院 | 精品国产乱码久久久久久久 | 免费观看三级毛片 | 97超级碰碰人国产在线观看 |