亚洲综合原千岁中文字幕_国产精品99久久久久久久vr_无码人妻aⅴ一区二区三区浪潮_成人h动漫精品一区二区三

主頁 > 知識庫 > 編寫高性能Lua代碼的方法

編寫高性能Lua代碼的方法

熱門標(biāo)簽:開通400電話申請流程 電腦外呼系統(tǒng)輻射大嗎 400手機電話免費辦理 智能語音電銷的機器人 武漢百應(yīng)人工智能電銷機器人 如何利用高德地圖標(biāo)注家 揚州電銷外呼系統(tǒng)軟件 上海企業(yè)外呼系統(tǒng)排名 百度地圖標(biāo)注位置網(wǎng)站

前言

Lua是一門以其性能著稱的腳本語言,被廣泛應(yīng)用在很多方面,尤其是游戲。像《魔獸世界》的插件,手機游戲《大掌門》《神曲》《迷失之地》等都是用Lua來寫的邏輯。

所以大部分時候我們不需要去考慮性能問題。Knuth有句名言:“過早優(yōu)化是萬惡之源”。其意思就是過早優(yōu)化是不必要的,會浪費大量時間,而且容易導(dǎo)致代碼混亂。

所以一個好的程序員在考慮優(yōu)化性能前必須問自己兩個問題:“我的程序真的需要優(yōu)化嗎?”。如果答案為是,那么再問自己:“優(yōu)化哪個部分?”。

我們不能靠臆想和憑空猜測來決定優(yōu)化哪個部分,代碼的運行效率必須是可測量的。我們需要借助于分析器來測定性能的瓶頸,然后著手優(yōu)化。優(yōu)化后,我們?nèi)匀灰柚诜治銎鱽頊y量所做的優(yōu)化是否真的有效。

我認為最好的方式是在首次編寫的時候按照最佳實踐去寫出高性能的代碼,而不是編寫了一堆垃圾代碼后,再考慮優(yōu)化。相信工作后大家都會對事后的優(yōu)化的繁瑣都深有體會。

一旦你決定編寫高性能的Lua代碼,下文將會指出在Lua中哪些代碼是可以優(yōu)化的,哪些代碼會是運行緩慢的,然后怎么去優(yōu)化它們。

使用local

在代碼運行前,Lua會把源碼預(yù)編譯成一種中間碼,類似于Java的虛擬機。這種格式然后會通過C的解釋器進行解釋,整個過程其實就是通過一個while循環(huán),里面有很多的switch...case語句,一個case對應(yīng)一條指令來解析。

自Lua 5.0之后,Lua采用了一種類似于寄存器的虛擬機模式。Lua用棧來儲存其寄存器。每一個活動的函數(shù),Lua都會其分配一個棧,這個棧用來儲存函數(shù)里的活動記錄。每一個函數(shù)的棧都可以儲存至多250個寄存器,因為棧的長度是用8個比特表示的。

有了這么多的寄存器,Lua的預(yù)編譯器能把所有的local變量儲存在其中。這就使得Lua在獲取local變量時其效率十分的高。

舉個栗子: 假設(shè)a和b為local變量,a = a + b的預(yù)編譯會產(chǎn)生一條指令:

復(fù)制代碼 代碼如下:

;a是寄存器0 b是寄存器1
ADD 0 0 1

但是若a和b都沒有聲明為local變量,則預(yù)編譯會產(chǎn)生如下指令:

復(fù)制代碼 代碼如下:

GETGLOBAL    0 0    ;get a
GETGLOBAL    1 1    ;get b
ADD          0 0 1  ;do add
SETGLOBAL    0 0    ;set a

所以你懂的:在寫Lua代碼時,你應(yīng)該盡量使用local變量。

以下是幾個對比測試,你可以復(fù)制代碼到你的編輯器中,進行測試。

復(fù)制代碼 代碼如下:

a = os.clock()
for i = 1,10000000 do
  local x = math.sin(i)
end
b = os.clock()
print(b-a) -- 1.113454

把math.sin賦給local變量sin:

復(fù)制代碼 代碼如下:

a = os.clock()
local sin = math.sin
for i = 1,10000000 do
  local x = sin(i)
end
b = os.clock()
print(b-a) --0.75951

直接使用math.sin,耗時1.11秒;使用local變量sin來保存math.sin,耗時0.76秒。可以獲得30%的效率提升!

關(guān)于表(table)

表在Lua中使用十分頻繁,因為表幾乎代替了Lua的所有容器。所以快速了解一下Lua底層是如何實現(xiàn)表,對我們編寫Lua代碼是有好處的。

Lua的表分為兩個部分:數(shù)組(array)部分和哈希(hash)部分。數(shù)組部分包含所有從1到n的整數(shù)鍵,其他的所有鍵都儲存在哈希部分中。

哈希部分其實就是一個哈希表,哈希表本質(zhì)是一個數(shù)組,它利用哈希算法將鍵轉(zhuǎn)化為數(shù)組下標(biāo),若下標(biāo)有沖突(即同一個下標(biāo)對應(yīng)了兩個不同的鍵),則它會將沖突的下標(biāo)上創(chuàng)建一個鏈表,將不同的鍵串在這個鏈表上,這種解決沖突的方法叫做:鏈地址法。

當(dāng)我們把一個新鍵值賦給表時,若數(shù)組和哈希表已經(jīng)滿了,則會觸發(fā)一個再哈希(rehash)。再哈希的代價是高昂的。首先會在內(nèi)存中分配一個新的長度的數(shù)組,然后將所有記錄再全部哈希一遍,將原來的記錄轉(zhuǎn)移到新數(shù)組中。新哈希表的長度是最接近于所有元素數(shù)目的2的乘方。

當(dāng)創(chuàng)建一個空表時,數(shù)組和哈希部分的長度都將初始化為0,即不會為它們初始化任何數(shù)組。讓我們來看下執(zhí)行下面這段代碼時在Lua中發(fā)生了什么:

復(fù)制代碼 代碼如下:

local a = {}
for i=1,3 do
    a[i] = true
end

最開始,Lua創(chuàng)建了一個空表a,在第一次迭代中,a[1] = true觸發(fā)了一次rehash,Lua將數(shù)組部分的長度設(shè)置為2^0,即1,哈希部分仍為空。在第二次迭代中,a[2] = true再次觸發(fā)了rehash,將數(shù)組部分長度設(shè)為2^1,即2。最后一次迭代,又觸發(fā)了一次rehash,將數(shù)組部分長度設(shè)為2^2,即4。

下面這段代碼:

復(fù)制代碼 代碼如下:

a = {}
a.x = 1; a.y = 2; a.z = 3

與上一段代碼類似,只是其觸發(fā)了三次表中哈希部分的rehash而已。

只有三個元素的表,會執(zhí)行三次rehash;然而有一百萬個元素的表僅僅只會執(zhí)行20次rehash而已,因為2^20 = 1048576 > 1000000。但是,如果你創(chuàng)建了非常多的長度很小的表(比如坐標(biāo)點:point = {x=0,y=0}),這可能會造成巨大的影響。

如果你有很多非常多的很小的表需要創(chuàng)建時,你可以將其預(yù)先填充以避免rehash。比如:{true,true,true},Lua知道這個表有三個元素,所以Lua直接創(chuàng)建了三個元素長度的數(shù)組。類似的,{x=1, y=2, z=3},Lua會在其哈希部分中創(chuàng)建長度為4的數(shù)組。

以下代碼執(zhí)行時間為1.53秒:

復(fù)制代碼 代碼如下:

a = os.clock()
for i = 1,2000000 do
    local a = {}
    a[1] = 1; a[2] = 2; a[3] = 3
end
b = os.clock()
print(b-a)  --1.528293

如果我們在創(chuàng)建表的時候就填充好它的大小,則只需要0.75秒,一倍的效率提升!

復(fù)制代碼 代碼如下:

a = os.clock()
for i = 1,2000000 do
    local a = {1,1,1}
    a[1] = 1; a[2] = 2; a[3] = 3
end
b = os.clock()
print(b-a)  --0.746453

所以,當(dāng)需要創(chuàng)建非常多的小size的表時,應(yīng)預(yù)先填充好表的大小。

關(guān)于字符串

與其他主流腳本語言不同的是,Lua在實現(xiàn)字符串類型有兩方面不同。

第一,所有的字符串在Lua中都只儲存一份拷貝。當(dāng)新字符串出現(xiàn)時,Lua檢查是否有其相同的拷貝,若沒有則創(chuàng)建它,否則,指向這個拷貝。這可以使得字符串比較和表索引變得相當(dāng)?shù)目欤驗楸容^字符串只需要檢查引用是否一致即可;但是這也降低了創(chuàng)建字符串時的效率,因為Lua需要去查找比較一遍。

第二,所有的字符串變量,只保存字符串引用,而不保存它的buffer。這使得字符串的賦值變得十分高效。例如在Perl中,$x = $y,會將$y的buffer整個的復(fù)制到$x的buffer中,當(dāng)字符串很長時,這個操作的代價將十分昂貴。而在Lua,同樣的賦值,只復(fù)制引用,十分的高效。

但是只保存引用會降低在字符串連接時的速度。在Perl中,$s = $s . 'x'和$s .= 'x'的效率差距驚人。前者,將會獲取整個$s的拷貝,并將'x'添加到它的末尾;而后者,將直接將'x'插入到$x的buffer末尾。

由于后者不需要進行拷貝,所以其效率和$s的長度無關(guān),因為十分高效。

在Lua中,并不支持第二種更快的操作。以下代碼將花費6.65秒:

復(fù)制代碼 代碼如下:

a = os.clock()
local s = ''
for i = 1,300000 do
    s = s .. 'a'
end
b = os.clock()
print(b-a)  --6.649481

我們可以用table來模擬buffer,下面的代碼只需花費0.72秒,9倍多的效率提升:

復(fù)制代碼 代碼如下:

a = os.clock()
local s = ''
local t = {}
for i = 1,300000 do
    t[#t + 1] = 'a'
end
s = table.concat( t, '')
b = os.clock()
print(b-a)  --0.07178

所以:在大字符串連接中,我們應(yīng)避免..。應(yīng)用table來模擬buffer,然后concat得到最終字符串。

3R原則

3R原則(the rules of 3R)是:減量化(reducing),再利用(reusing)和再循環(huán)(recycling)三種原則的簡稱。

3R原則本是循環(huán)經(jīng)濟和環(huán)保的原則,但是其同樣適用于Lua。

Reducing

有許多辦法能夠避免創(chuàng)建新對象和節(jié)約內(nèi)存。例如:如果你的程序中使用了太多的表,你可以考慮換一種數(shù)據(jù)結(jié)構(gòu)來表示。

舉個栗子。 假設(shè)你的程序中有多邊形這個類型,你用一個表來儲存多邊形的頂點:

復(fù)制代碼 代碼如下:

polyline = {
    { x = 1.1, y = 2.9 },
    { x = 1.1, y = 3.7 },
    { x = 4.6, y = 5.2 },
    ...
}

以上的數(shù)據(jù)結(jié)構(gòu)十分自然,便于理解。但是每一個頂點都需要一個哈希部分來儲存。如果放置在數(shù)組部分中,則會減少內(nèi)存的占用:

復(fù)制代碼 代碼如下:

polyline = {
    { 1.1, 2.9 },
    { 1.1, 3.7 },
    { 4.6, 5.2 },
    ...
}

一百萬個頂點時,內(nèi)存將會由153.3MB減少到107.6MB,但是代價是代碼的可讀性降低了。

最變態(tài)的方法是:

復(fù)制代碼 代碼如下:

polyline = {
    x = {1.1, 1.1, 4.6, ...},
    y = {2.9, 3.7, 5.2, ...}
}

一百萬個頂點,內(nèi)存將只占用32MB,相當(dāng)于原來的1/5。你需要在性能和代碼可讀性之間做出取舍。

在循環(huán)中,我們更需要注意實例的創(chuàng)建。

復(fù)制代碼 代碼如下:

for i=1,n do
    local t = {1,2,3,'hi'}
    --執(zhí)行邏輯,但t不更改
    ...
end

我們應(yīng)該把在循環(huán)中不變的東西放到循環(huán)外來創(chuàng)建:

復(fù)制代碼 代碼如下:

local t = {1,2,3,'hi'}
for i=1,n do
    --執(zhí)行邏輯,但t不更改
    ...
end

Reusing

如果無法避免創(chuàng)建新對象,我們需要考慮重用舊對象。

考慮下面這段代碼:

復(fù)制代碼 代碼如下:

local t = {}
for i = 1970, 2000 do
    t[i] = os.time({year = i, month = 6, day = 14})
end

在每次循環(huán)迭代中,都會創(chuàng)建一個新表{year = i, month = 6, day = 14},但是只有year是變量。

下面這段代碼重用了表:

復(fù)制代碼 代碼如下:

local t = {}
local aux = {year = nil, month = 6, day = 14}
for i = 1970, 2000 do
    aux.year = i;
    t[i] = os.time(aux)
end

另一種方式的重用,則是在于緩存之前計算的內(nèi)容,以避免后續(xù)的重復(fù)計算。后續(xù)遇到相同的情況時,則可以直接查表取出。這種方式實際就是動態(tài)規(guī)劃效率高的原因所在,其本質(zhì)是用空間換時間。

Recycling

Lua自帶垃圾回收器,所以我們一般不需要考慮垃圾回收的問題。

了解Lua的垃圾回收能使得我們編程的自由度更大。

Lua的垃圾回收器是一個增量運行的機制。即回收分成許多小步驟(增量的)來進行。

頻繁的垃圾回收可能會降低程序的運行效率。

我們可以通過Lua的collectgarbage函數(shù)來控制垃圾回收器。

collectgarbage函數(shù)提供了多項功能:停止垃圾回收,重啟垃圾回收,強制執(zhí)行一次回收循環(huán),強制執(zhí)行一步垃圾回收,獲取Lua占用的內(nèi)存,以及兩個影響垃圾回收頻率和步幅的參數(shù)。

對于批處理的Lua程序來說,停止垃圾回收collectgarbage("stop")會提高效率,因為批處理程序在結(jié)束時,內(nèi)存將全部被釋放。

對于垃圾回收器的步幅來說,實際上很難一概而論。更快幅度的垃圾回收會消耗更多CPU,但會釋放更多內(nèi)存,從而也降低了CPU的分頁時間。只有小心的試驗,我們才知道哪種方式更適合。

結(jié)語

我們應(yīng)該在寫代碼時,按照高標(biāo)準(zhǔn)去寫,盡量避免在事后進行優(yōu)化。

如果真的有性能問題,我們需要用工具量化效率,找到瓶頸,然后針對其優(yōu)化。當(dāng)然優(yōu)化過后需要再次測量,查看是否優(yōu)化成功。

在優(yōu)化中,我們會面臨很多選擇:代碼可讀性和運行效率,CPU換內(nèi)存,內(nèi)存換CPU等等。需要根據(jù)實際情況進行不斷試驗,來找到最終的平衡點。

最后,有兩個終極武器:

第一、使用LuaJIT,LuaJIT可以使你在不修改代碼的情況下獲得平均約5倍的加速。查看LuaJIT在x86/x64下的性能提升比。

第二、將瓶頸部分用C/C++來寫。因為Lua和C的天生近親關(guān)系,使得Lua和C可以混合編程。但是C和Lua之間的通訊會抵消掉一部分C帶來的優(yōu)勢。

注意:這兩者并不是兼容的,你用C改寫的Lua代碼越多,LuaJIT所帶來的優(yōu)化幅度就越小。

聲明

這篇文章是基于Lua語言的創(chuàng)造者Roberto Ierusalimschy在Lua Programming Gems 中的Lua Performance Tips翻譯改寫而來。本文沒有直譯,做了許多刪節(jié),可以視為一份筆記。

感謝Roberto在Lua上的辛勤勞動和付出!

標(biāo)簽:延邊 宜賓 嘉峪關(guān) 黑龍江 新余 武漢 張掖 江西

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《編寫高性能Lua代碼的方法》,本文關(guān)鍵詞  編寫,高性能,Lua,代碼,的,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《編寫高性能Lua代碼的方法》相關(guān)的同類信息!
  • 本頁收集關(guān)于編寫高性能Lua代碼的方法的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    日韩在线观看免费| 欧美激情在线精品video| 99久久精品国产国产毛片| 国产视频一区在线| 日本特黄一级| 国产亚洲精品aaa大片| 国产成人啪精品视频免费软件| 精品久久久久久免费影院| 毛片的网站| 国产伦精品一区三区视频| 国产国语在线播放视频| 亚洲第一视频在线播放| 香蕉视频久久| 国产精品123| 日韩女人做爰大片| 欧美国产日韩久久久| 美女免费毛片| 午夜在线亚洲男人午在线| 日韩综合| 国产麻豆精品免费密入口| 亚洲 国产精品 日韩| 日韩中文字幕在线观看视频| 一级女性全黄生活片免费| 成人影院一区二区三区| 亚洲不卡一区二区三区在线| 亚洲www美色| 国产不卡高清| 精品国产亚洲人成在线| 国产高清在线精品一区二区| 日本在线www| 一级女性大黄生活片免费| 精品在线观看一区| 一 级 黄 中国色 片| 日本特黄一级| 久久精品免视看国产明星| 久久精品成人一区二区三区| 国产伦精品一区三区视频| 国产91精品一区| 国产精品123| 99久久精品国产国产毛片| 国产极品精频在线观看| 天天做人人爱夜夜爽2020毛片| 久久精品道一区二区三区| 日韩一级黄色大片| 日韩女人做爰大片| 四虎影视久久久| 99色视频在线| 国产伦精品一区二区三区无广告 | 国产国语在线播放视频| 欧美1区| 成人免费观看的视频黄页| 九九干| 四虎影视库| 毛片的网站| 麻豆网站在线看| 欧美电影免费看大全| 亚洲第一视频在线播放| 日韩av成人| 国产成人啪精品视频免费软件| 深夜做爰性大片中文| 韩国三级香港三级日本三级la | 国产亚洲精品aaa大片| 国产一区精品| 国产a一级| 久久99爰这里有精品国产| 国产伦精品一区三区视频| 尤物视频网站在线观看| 日韩一级黄色大片| 九九久久国产精品| 日本在线不卡视频| 99久久网站| 欧美一级视| 亚洲第一页色| 午夜激情视频在线观看| 欧美激情在线精品video| 久久国产影院| 91麻豆高清国产在线播放| 97视频免费在线观看| 九九精品影院| 九九免费精品视频| 欧美激情一区二区三区视频| 青青青草影院| 91麻豆精品国产综合久久久| 欧美爱色| 精品久久久久久免费影院| 日本特黄特黄aaaaa大片| 91麻豆高清国产在线播放| 国产精品自拍亚洲| 香蕉视频一级| 国产伦精品一区三区视频| 中文字幕一区二区三区 精品| 99久久网站| 一级女性大黄生活片免费| 香蕉视频久久| 亚洲 男人 天堂| 日韩中文字幕一区二区不卡| 精品视频一区二区三区| 精品久久久久久免费影院| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 精品在线观看一区| 亚洲精品中文一区不卡| 免费一级片在线观看| 国产不卡在线看| 久久国产精品自由自在| 日本在线不卡视频| 精品视频在线观看一区二区三区| 日韩av成人| 精品久久久久久中文| 久久国产精品自由自在| 999久久66久6只有精品| 国产福利免费观看| 国产成人啪精品视频免费软件| 国产原创中文字幕| 韩国毛片基地| 黄视频网站免费观看| 青青久久网| 毛片的网站| 国产综合成人观看在线| 成人高清视频免费观看| 91麻豆tv| 国产伦精品一区三区视频| 亚洲第一页色| 九九干| 日韩一级黄色大片| 国产不卡高清| 97视频免费在线观看| 日本在线www| 免费一级片在线观看| 成人免费福利片在线观看| 一级女性全黄生活片免费| 一本高清在线| 欧美另类videosbestsex久久| 香蕉视频一级| 国产亚洲精品aaa大片| 可以在线看黄的网站| 天天色色网| 麻豆午夜视频| 免费国产在线观看不卡| 韩国三级香港三级日本三级la | 美女免费毛片| 久久精品成人一区二区三区| 精品国产三级a∨在线观看| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 国产网站在线| 一级毛片看真人在线视频| 成人高清视频免费观看| 91麻豆国产福利精品| 久久久久久久网| 成人免费网站视频ww| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 四虎影视库| 成人免费观看视频| 韩国三级视频网站| 欧美激情一区二区三区在线播放| 91麻豆国产福利精品| 你懂的国产精品| 久久精品免视看国产明星| 日韩av成人| 国产福利免费观看| 九九九国产| 日本在线不卡视频| 免费毛片播放| 午夜在线亚洲男人午在线| 91麻豆精品国产自产在线| 二级片在线观看| 一级女性大黄生活片免费| 久久国产精品只做精品| 精品久久久久久免费影院| 精品国产一级毛片| 欧美一级视频免费| 亚洲 国产精品 日韩| 国产精品免费精品自在线观看| 精品国产一级毛片| 国产麻豆精品视频| 日韩av成人| 亚洲 男人 天堂| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 一本高清在线| 青青久久国产成人免费网站| a级毛片免费全部播放| 亚洲 男人 天堂| 999久久久免费精品国产牛牛| 亚洲不卡一区二区三区在线| 久久精品道一区二区三区| 免费国产在线观看不卡| 999久久久免费精品国产牛牛| 一级女性全黄生活片免费| 国产a视频| 可以在线看黄的网站| 一本高清在线| 国产高清视频免费观看| 国产麻豆精品hdvideoss| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 黄色福利片| 99热精品一区| 九九久久99| 99热视热频这里只有精品| 毛片的网站| 精品国产三级a∨在线观看| 久久精品免视看国产明星|