探索關(guān)于Lua自動綁定系統(tǒng)問題
關(guān)于Lua自動綁定系統(tǒng)問是本文要介紹的內(nèi)容,因為游戲內(nèi)容增長的速度遠(yuǎn)遠(yuǎn)超過了以前,所以程序員已經(jīng)不能夠完全控制所有代碼行為了。他們需要其他游戲開發(fā)者的幫助。腳本語言在游戲里已經(jīng)被用了幾十年,但是現(xiàn)在的游戲機(jī)更能抓住它們的優(yōu)點來打打提高玩家的體驗。
本精粹著重說明Lua語言的綁定實現(xiàn)。這門技術(shù)能夠讓程序員將它們的C++類暴露給Lua,但不需要了解這個系統(tǒng)。這里介紹的工具不僅能用在C++語言里,對其他語言也有效。這種設(shè)計的核心思想是易用性、效率、內(nèi)存占用和多線程的設(shè)計目標(biāo)來驅(qū)動的。
1、介紹
在本精粹里介紹的綁定玉虛在Lua腳本里建立、訪問和使用C++對象。舉例來說,在一個singleton的類WORLD里保存了一個ENTITY類型的實力列表,下面的腳本就能夠用來設(shè)置玩家的生命值。
- local entity = WORLD:GetEntity("player")
- entity:SetHealth(50)
在這個例子里用到的綁定是由如下的聲明定義的。
- //in.h and class definition
- SCRIPTABLE_DefineClass(WORLD)
- //in.cpp
- SCRIPTABLE_Class(WORLD)
- {
- SCRIPTABLE_ResultMethod1(GetEntity,ENTITY,std::string)
- }
綁定一個類就這么簡單。不需要做其他任何步驟,允許程序員把C++類和自己的函數(shù)暴露給Lua,就這么簡單。
2、特性
這個形同的設(shè)計目標(biāo)如下:
低內(nèi)存消耗;
高效率綁定;
支持C++的繼承;
使用方便;
在腳本與腳本之間保證線程使用的安全。
3、函數(shù)的綁定
Lua需要一個特別借口來綁定函數(shù)。被綁定的函數(shù)必須是下面代碼里定義的類型。Lua的綁定是基于堆棧的,lua_State包含了所有傳到那個函數(shù)里去的參數(shù)。這些參數(shù)必須通過使用堆棧索引號 lua_to*來收集。在這個例子里,這個函數(shù)接受到的第一個參數(shù)是字符串,第二個參數(shù)是數(shù)字,返回值也是數(shù)字。關(guān)于C函數(shù)的綁定,你可以在Lua手冊[Ierusalimschy06]里找到更多的相關(guān)信息。C函數(shù)的綁定是Lua綁定到C/C++的唯一一個方法,也是這個系統(tǒng)的基礎(chǔ)。
- int bingding_method(lua_State* state)
- {
- const char* some_string;
- double some_number,another_number;
- some_string = lua_tostring(state,1);
- some_number = lua_tonumber(state,2);
- //在這里可以設(shè)置返回值,或者做點你需要做的事情
- //另外一個數(shù)字
- lua_pushnumber(state,another_number);
- return 1;//假如說我們返回1
- }
4、在Lua里的面向?qū)ο?/p>
Lua是使用廣泛的一種編程語言,功能也很強(qiáng)大。本精粹介紹了如何把Lua變成一個面向?qū)ο蟮恼Z言。為了幫助大家用好這個功能,Lua的作者們定義了一系列的工具來給大家提供語法幫助(syntactic sugar)。下面列出我們在本系統(tǒng)里使用的一個。在這段代碼里,the_object是一個初始的變量,這段代碼模擬了this_call的函數(shù)返回。
- the_object:Test(5)==the_object["Test"]( the_object,5)
面向?qū)ο蟮姆椒梢允怯眠@種語法來實現(xiàn)。對象被當(dāng)做關(guān)系數(shù)組,用函數(shù)名來進(jìn)行索引,返回的就是你需要調(diào)用的函數(shù)。Lua里有一個機(jī)制,允許通過使用元表(metatable)來讓任何類型的變量對一個數(shù)組進(jìn)行交互(這事Lua5.1的特性。Lua5.0中只有表和用戶數(shù)據(jù)對象才有元表)。元表是Lua里的表,這個表被分配給一個對象,這個對象包含一些特殊的字段:_index, _newindex等([Ierusalimschy06])。在那些特殊字段里設(shè)置的函數(shù)會根據(jù)情況來被調(diào)用。當(dāng)一個對象被訪問時,采取的如果是數(shù)組形式的訪問方式,_index就被調(diào)用了。下面的代碼說明了如何講一個元表設(shè)置到一個對象上。
- metatable = {}
- metatable._ _index = function(table,key) return key end
- setmetatable(object, metatable)
- test_return = object["Test"] --在元表里面調(diào)用_index函數(shù)
Lua的內(nèi)部函數(shù)類型有數(shù)字(double 或者float)、string、table、nil、function(Lua或C)、thread和(輕)用戶數(shù)據(jù)。我們采用最后一個類型在Lua里保存對象。輕用戶數(shù)據(jù)和用戶數(shù)據(jù)稍微有點不一樣。第二種完全是Lua對象,可以擁有一個元表。
5、在Lua里綁定C++對象
綁定需要幾個機(jī)制:在Lua重新描述C++對象、綁定函數(shù)的保存,最后是每個C++對象綁定數(shù)據(jù)的注冊。在本精粹里,我們先把整體的技術(shù)介紹給大家,稍后我們會講解一些特別的例子。
綁定數(shù)據(jù)結(jié)構(gòu)
在已經(jīng)存在的現(xiàn)實中,綁定是直接保存在Lua里面的,而相關(guān)的綁定數(shù)據(jù)就幫存在每個腳本里。但是如果系統(tǒng)必須支持的腳本數(shù)量很大,那么綁定數(shù)據(jù)就回不必要的被重復(fù)保存。為了避免這樣的情況發(fā)生,我們決定把綁定數(shù)據(jù)保存在C++中一個叫SCRIPTABLE_BINDING_DATA的類里。每個綁定的類都會被分配到一個索引值。
SCRIPTABLE_BINDING_DATA里包含一個記錄雷鳴和類的索引號的映射,它被保存在CLassIndexTable里。然后每個類都有個映射,記錄每個函數(shù)名和對應(yīng)的綁定函數(shù)。MethodTable是這種映射的重組,可以根據(jù)ClassIndex Table里的值進(jìn)行索引。因為delete操作符是沒有名字的,所以它的綁定被存放在單獨的數(shù)組里,這個數(shù)組叫做Delete Table。最后,Parent Table保存了每個類的弗雷的索引。如果某個類是沒有父類的,那么Parent Table的入口就被設(shè)置了 -1
在本書附帶光盤中,你可以找到一系列的副主函數(shù),這些輔助函數(shù)能夠讓你訪問這些映射。你可以在 scriptable_bingding_data.h文件里找到它們。
- class SCRIPTABLE_BINDING_DATA
- {
- typedef int(* BINDING_FUNCTION) (lua_State *);
- std::map<std:string,int>
- ClassIndexTable;
- std::vector<std::map<std::string,BINDING_FUNCTION>*>
- MethodTable;
- std::vector<BINDING_FUNTION>
- DeleteTable;
- std::vector<int>;
- ParentTable;
- };
指向這個綁定數(shù)據(jù)的指針和之歌類的索引被存放在lua_State里。這個數(shù)據(jù)的空間是由luaconf.h里的LUAI_EXTRASPACE常量分配的。
小結(jié):探索關(guān)于Lua自動綁定系統(tǒng)問題的內(nèi)容介紹完了,希望通過本文的學(xué)習(xí)能對你有所幫助!