Lua和C#調用探秘


轉載請標明出處:http://www.cnblogs.com/zblade/

在實際的項目中,大部分業務邏輯 程序員只需要負責lua層編寫邏輯即可,或者在c#層添加一些靜態函數,供lua層調用。那么對於具體的相互之間的交互,又是如何進行的?本文就寫一寫個人的一些探究筆記吧。本文會寫很多代碼,我就用截圖來展示吧,編輯寫代碼不大方便,有點蛋疼~

一、c#函數的導出

我就從外部接口開始理一遍整體思路吧,想了一下,還是從代碼思路來解釋比較容易。

首先我們的工程中都會有一個slua的導出接口:

這樣的一個接口,是用來將UnityEngine中的類導出的實現API,其整體的思路是:

1)首先加載UnityEngine這個程序集:

Assembly assembly = Assembly.Load("UnityEngine") 

2)然后獲取資中的可導出類型:

Type[] types = assembly.GetExportedTypes(); 

3) 做一次過濾,主要是對於某些需要導出的類和不需要導出的類做一次過濾剔除和添加,這個不同項目不一樣,不做展示;

4)將這些過濾后的類型,逐個做一次導出,比如相機類,可以導出為:

5)將這些導出的類Lua_xxx合並在一起作為一個Bind, 提供一個靜態獲取方法GetBindList()

這是第一步,完成對c#和unity中的方法導出,將每個不同程序集中的類中的方法和屬性都暴露出來,做一個導出。

二、導出的c#文件的注冊到Lua虛擬機中

這部分需要結合游戲的啟動來理解,在游戲的啟動時刻,我們都會啟動一個Lua的虛擬機,比如這樣:

在啟動虛擬機后,需要執行虛擬機的Init操作:

m_LuaSvr.init(xxxx) 

在這個函數中,執行Bind的操作:doBind

其中的關鍵操作為collectBindInfo, 這個函數分為2部分:

1)獲取當前程序集,以及程序集中設置為LuaBinderAttribute的類型:

2)根據獲取的類型,逐個反射執行第一部分最后的GetBindList函數:

 

這樣通過c#的反射,就可以動態的獲取前面導出的所有LuaXXX類文件了,回到Bind操作,對於這些獲取的Lua_XXX文件,執行Lua虛擬機的注冊操作:

action(L) 

也就是導出文件中的reg操作:

看看其操作,首先是newtable的操作:

創建2個table,分別用來做static和instance的填充,然后填充的操作addMember:

對於不同的參數,會重載不同的addMember操作,這兒就舉例一個,pushValue就是將func注冊到該table中:

LuaDll.lua_pushcclosure(L, function, 0) 

就是將該函數填充到lua表中,可以通過key名的查找來獲取該函數,從而執行相關的調用。

最后會在該reg操作中為該類創建一個metatable

回到最初的,不斷的循環執行,就可以加載整個c#相關導出類到Lua虛擬機中

總結:到現在為止,可以知道整個c#函數在導出過程中的操作,在啟動時候如何通過程序集和反射來實現動態的加載,最后Lua的虛擬機中都會注冊前面導出的類文件的相關函數和屬性。

而我們已經知道,lua文件在執行的時候,是會編譯成字節碼在lua的虛擬機中執行的,這樣lua的字節碼和c#的導出文件,都在同一個環境中執行,調用pcall就可以相互的執行和調用了。

寫這篇文章是基於偶然翻看到老外寫的一個在unity中用c++做腳本來編寫游戲邏輯,並且實現了c#和c++之間的相互交互調用,所以我也翻看了一下c#是如何實現的,當然寫的比較簡陋,還有很多細節需要推敲,大家可以翻看自己的項目代碼,留言討論


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM