C#泛型方法在lua中表示的一種設(shè)計(jì)詳解
前言
在進(jìn)行l(wèi)ua方法注冊(cè)的時(shí)候, 大多數(shù)解決方案直接否定了泛型方法, 因?yàn)樵趌ua側(cè)難以表達(dá)出泛型, 以及l(fā)ua的函數(shù)重載問(wèn)題,
函數(shù)重載問(wèn)題可以通過(guò)一些特殊方法解決, 而泛型問(wèn)題是主要問(wèn)題, 以Unity + Slua的情況來(lái)說(shuō)
比如下面的類:
public class Foo { public static void GetTypeName(System.Type type) { Debug.Log(type.Name); } public static void GetTypeName<T>() { Debug.Log(typeof(T).Name); } }
一般只會(huì)生成 GetTypeName(System.Type type)
的注冊(cè)方法.
那么泛型的方法在Lua那邊該怎樣注冊(cè)才能讓這個(gè)調(diào)用能夠?qū)崿F(xiàn)呢? 一般來(lái)說(shuō)我們調(diào)用泛型方法必須在寫代碼的時(shí)候就確定, 像這樣:
Foo.GetTypeName<int>(); // 輸出 Int32
而lua并不能這樣約束, 它的調(diào)用必須還是非泛型的才可以, 這是第一個(gè)問(wèn)題, 而第二個(gè)問(wèn)題是lua那邊怎樣寫? 我們希望它的寫法能跟C#保持
一致, 或者相似吧, 讓人看起來(lái)容易明白, 可是lua中中括號(hào)是大于小于號(hào), 不能這樣寫, 想想有沒(méi)有什么辦法
因?yàn)樵趌ua中是沒(méi)有類型的, 類型必須來(lái)自C#, 所以只能將泛型作為非泛型方法才能使用, 如果讓函數(shù)進(jìn)行一次退化和封裝, 像下面這樣
-- 先將C# 的typeof注冊(cè)成全局函數(shù), 注冊(cè)System.Int32命名為int local Foo = {} Foo.GetTypeName = function(type) return function() print(type.Name) end end
Foo.GetTypeName(typeof(int))(); -- lua Foo.GetTypeName<typeof(int)>(); // C#
這樣寫的話, 除了尖括號(hào), 基本就能兩邊一致了對(duì)吧, 運(yùn)行結(jié)果也是一樣的
/*至于怎樣注冊(cè)typeof(int)*/ // 在LuaState的Init中注冊(cè)個(gè)全局函數(shù)[MonoPInvokeCallbackAttribute(typeof(LuaCSFunction))] internal static int getType(IntPtr L) { System.Type type = null; LuaObject.checkType(L, 1, out type); LuaObject.pushObject(L, type); return 1; } // 在LuaState的Init中自己注冊(cè)咯LuaDLL.lua_pushcfunction(L, getType);LuaDLL.lua_setglobal(L, "typeof");
// CustomExport.OnAddCustomClass 中添加類型別名 add(typeof(System.Int32), "int"); // int
只是這里lua的函數(shù)沒(méi)有進(jìn)行C#那邊的調(diào)用啊, 下一步就來(lái)看看有沒(méi)有什么辦法來(lái)實(shí)現(xiàn)調(diào)用.
如果通過(guò)自動(dòng)注冊(cè)的話, Foo應(yīng)該是一個(gè)已經(jīng)注冊(cè)的類型.
[SLua.CustomLuaClass] public class Foo
并且有元表, 元表里面有非泛型的GetTypeName方法了. 現(xiàn)在先不要去動(dòng)元表,
直接注冊(cè)這個(gè)到Table里面, 因?yàn)槿绻鸗able里面有值的話, 就不會(huì)去查詢?cè)砹?/p>
import "Foo"; Foo.GetTypeName(typeof(int)); // 輸出 Int32 rawset(Foo, "GetTypeName", function(type) return function() local mt = getmetatable(Foo) local func = rawget(mt,"GetTypeName"); func(type) end end) Foo.GetTypeName(typeof(int))(); // 輸出 Int32 -- 注意返回了function然后再次調(diào)用
這個(gè)方法比較流氓, 因?yàn)橹苯幽J(rèn)了有非泛型函數(shù), 并且覆蓋了元表的非泛型方法, 不可取的.
要繼續(xù)的話, 首先來(lái)看看一個(gè)泛型方法怎樣通過(guò)Type方法進(jìn)行調(diào)用的:
var methods = typeof(Foo).GetMethods(BindingFlags.Public | BindingFlags.Static | BindingFlags.InvokeMethod); foreach(var method in methods) { if(method.IsGenericMethod) { var paramters = method.GetParameters(); if(paramters == null || paramters.Length == 0) { var genericMethod = method.MakeGenericMethod(new Type[] { typeof(int) }); if(genericMethod != null) { genericMethod.Invoke(null, null); // 輸出 Int32 break; } } } }
當(dāng)然是反射啦, 這樣就能讓泛型方法退化為非泛型了, 雖然是一個(gè)緩慢的反射, 不過(guò)時(shí)間基本只花費(fèi)在Invoke上, 問(wèn)題還不大.
剩下的問(wèn)題是重載了, 有非泛型和泛型的兩個(gè)同名函數(shù), 為了測(cè)試我先刪除掉非泛型,
[SLua.CustomLuaClass] public class Foo { //public static void GetTypeName(System.Type type) //{ // Debug.Log(type.Name); //} public static void GetTypeName<T>() { Debug.Log(typeof(T).Name); } }
生成的lua注冊(cè)代碼也要修改一下
System.Type a1; checkType(l,1,out a1); Foo.GetTypeName(a1); // 它完了 pushValue(l,true);
改成
System.Type a1; checkType(l,1,out a1); var methods = typeof(Foo).GetMethods(System.Reflection. BindingFlags.Public | System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.InvokeMethod); foreach(var method in methods) { if(method.IsGenericMethod) { var paramters = method.GetParameters(); if(paramters == null || paramters.Length == 0) { var genericMethod = method.MakeGenericMethod(new Type[] { typeof(int) }); if(genericMethod != null) { genericMethod.Invoke(null, null); break; } } } } pushValue(l,true);
試試運(yùn)行一下看看, 輸出 Int32 看來(lái)沒(méi)有問(wèn)題, 問(wèn)題是在Lua那邊還是需要手動(dòng)封裝了一遍:
rawset(Foo, "GetTypeName", function(type) local mt = getmetatable(Foo) local func = rawget(mt,"GetTypeName"); func(type) end) -- 問(wèn)題是, 不進(jìn)行一次rawset無(wú)法得到泛型寫法 Foo.GetTypeName(typeof(int)); // 輸出 Int32 -- Table方法
到這里, 基本就可以得出結(jié)論了,
一. 在lua中可以通過(guò)封裝(閉包)的方式接近C#的泛型的寫法, 差別只是一個(gè)中括號(hào)和小括號(hào)
Foo.GetTypeName(typeof(int))(); -- lua Foo.GetTypeName<typeof(int)>(); // C#
然而過(guò)程異常復(fù)雜, 比如上述代碼中的rawset過(guò)程需要在C#的注冊(cè)代碼中進(jìn)行實(shí)現(xiàn), 而在調(diào)用的地方需要通過(guò)反射, 并且在lua側(cè)需要解決函數(shù)重載的問(wèn)題,
上面的例子直接做了覆蓋. 就無(wú)法正常訪問(wèn)非泛型方法函數(shù)了.
二. 既然泛型方法可以退化為非泛型, 那么可以直接檢測(cè)有沒(méi)有同名的且同參數(shù)的非泛型函數(shù), 如果沒(méi)有就把泛型方法的非泛型版添加到注冊(cè)函數(shù)中即可.
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
C#中的自動(dòng)類型轉(zhuǎn)換和強(qiáng)制類型轉(zhuǎn)換
這篇文章主要介紹了C#中的自動(dòng)類型轉(zhuǎn)換和強(qiáng)制類型轉(zhuǎn)換,非常不錯(cuò),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下2019-08-08C#設(shè)置PDF表單不可編輯以及提取表單數(shù)據(jù)的操作
PDF表單是PDF中的可編輯區(qū)域,允許用戶填寫指定信息,當(dāng)表單填寫完成后,有時(shí)候我們可能需要將其設(shè)置為不可編輯,以保護(hù)表單內(nèi)容的完整性和可靠性,本文將給大家介紹C#設(shè)置PDF表單不可編輯以及提取表單數(shù)據(jù)的操作,需要的朋友可以參考下2024-06-06解析在內(nèi)部循環(huán)中Continue外部循環(huán)的使用詳解
本篇文章是對(duì)在內(nèi)部循環(huán)中Continue外部循環(huán)的使用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05AOP從靜態(tài)代理到動(dòng)態(tài)代理(Emit實(shí)現(xiàn))詳解
AOP為Aspect Oriented Programming的縮寫,意思是面向切面編程的技術(shù)。下面這篇文章主要給大家介紹了關(guān)于AOP從靜態(tài)代理到動(dòng)態(tài)代理(Emit實(shí)現(xiàn))的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2018-09-09C#取得Web程序和非Web程序的根目錄的N種取法總結(jié)
C#取得Web程序和非Web程序的根目錄的N種取法,方便大家知道,有更好的方法,請(qǐng)說(shuō)明2008-03-03C#與C++動(dòng)態(tài)鏈接庫(kù)DLL參數(shù)互傳方式
這篇文章主要介紹了C#與C++動(dòng)態(tài)鏈接庫(kù)DLL參數(shù)互傳方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11C#如何通過(guò)QQ郵件發(fā)送驗(yàn)證碼到指定郵箱
在程序設(shè)計(jì)中發(fā)送驗(yàn)證碼是常見的一個(gè)功能,用戶在注冊(cè)賬號(hào)時(shí)或忘記密碼后通常需要發(fā)送驗(yàn)證碼到手機(jī)短信或郵箱來(lái)驗(yàn)證身份,這篇文章主要給大家介紹了關(guān)于C#如何通過(guò)QQ郵件發(fā)送驗(yàn)證碼到指定郵箱的相關(guān)資料,需要的朋友可以參考下2024-01-01