青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

woaidongmao

文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
數據加載中……

實現自己的LUA綁定器-一個模板編程挑戰

實現LUA綁定器

author : Kevin Lynx

Preface


    當LUA腳本調用我們注冊的C函數時,我們需要逐個地從LUA棧里取出調用參數,當函數返回時,又需要一個一個地往LUA
棧壓入返回值,并且我們注冊的函數只能是int()(lua_State*)類型。這很不方便,對于上層程序員來說更不方便。
    因此我們要做的是,實現一個綁定器,可以把任意prototype的函數綁定到LUA腳本當中,并且封裝取參數和壓返回值時
的諸多細節。
    確實,世界上已經有很多庫做了這件事情。但是,我們這里的需求很簡單,我們只需要綁定函數,而不需要綁定C++類之
類的東西,自己實現的才是輕量級的。

What we usually do

    先看下我們平時是怎么做這些事的。首先注冊函數很簡單:

    lua_pushcfunction( L, to_string );
    lua_setglobal( L,
"tostr"
);


    然后是func的具體處理:

    /** 假設to_string在腳本中的原型是: string ()( number ) */
   
static int to_string( lua_State *L )
   
{
       
static char buf[512
];
       
/* 從LUA棧中取參數 */

       
int n = (int)lua_tonumber( L, -1 );
        sprintf( buf,
"%d"
, n );
       
/* 壓入返回值 */

        lua_pushstring( L, buf );
       
return 1;
    }


    這是個簡單的例子,目的是展示我之前說的局限性,以及,恩,丑陋性。


How

    要讓事情變得優美,我們就得隱藏丑陋。
    首先,我們看看如何改進to_string的處理,使其看起來干凈。最直接也是最通用的做法是,我們自己做一個粘合層,
充當LUA與應用層之間的粘合劑。也就是說,LUA直接回調的不再直接是應用層的函數,而是我們實現的這一層中的函數,
我們的函數整理調用參數,然后回調到上層函數,上層返回后,我們收集上層的返回值,然后整理給LUA,最后返回。
    這就是思路,具體實現時更為有趣。

Implementing...

    直覺告訴我,我需要使用C++模板來實現。模板和宏都是個好東西,因為它們是泛性的,它們給程序員帶來自動性。
    另一個直覺告訴我,盡量不要讓上層保存任何東西。通過模板的實例化,編譯器已經為我們添加了很多東西,我也不
想讓上層理會我太多。
    因為,我們至少需要保存上層的函數指針(我們暫時只考慮C式的函數),我們至少還需要一個粘合層函數用以被LUA
直接回調,所以,我得到了以下類模板:

    template <typename Prototype>
   
class lua_binder
   
{
   
public
:
        typedef Prototype func_type;
   
public
:
       
static int adapter( lua_State *
L )
       
{
           
return 0
;
        }
 

   
public
:
       
static
func_type _func;
    }
;
    template
<typename Prototype> typename lua_binder<Prototype>::func_type lua_binder<Prototype>::_func = 0
;


    這樣,泛化了Prototype后,lua_binder可以保存任意原型的函數指針。例如:

 

typedef lua_binder<const char*(int)> binder_type;

  
    借助于模板技術,即使上層只是這樣一個簡單的看似不會產生任何代碼的typedef,實際上也會產生出一個static的
函數指針變量:_func。

    這個時候,我們也該考慮下注冊函數部分了。注意,事實上我們總共需要干兩件事:封裝粘合層函數、封裝注冊函數
部分。同樣,我們得到一個最直觀的注冊函數模板:

 

    template <typename binder_type>
   
void lua_bind( lua_State *L, typename binder_type::func_type func, const char *name )
   
{
        binder_type::_func
=
func;
        lua_pushcfunction( L, binder_type::adapter );
        lua_setgloabl( L, name );
    }


    為什么模板參數是binder_type而不是Prototype?(最直接的想法可能會想到Prototype)因為我們需要獲取func_type
以及最重要的:設置_func的值!綜合起來,lua_bind函數主要作用就是接受用戶層函數指針,并相應的將粘合層函數注冊
到LUA中。注意,lua_pushcfunction注冊的是binder_type::adapter函數。

    那么,理論上,我們現在可以這樣注冊一個函數:

    typedef lua_binder<const char*(_cdecl*)(int)> binder_type;
    lua_bind
<binder_type>( L, to_string, "tostr"
);


    (這個時候to_string為:const char* to_string( int ) )


處理函數參數的個數

    事情遠沒有我們想象的那么簡單。adapter函數中毫無實現,重要的是,該如何去實現?我們面對的首個問題是:上層
函數參數個數不一樣,那么我們的adapter該調用多少次lua_to*去從LUA棧中獲取參數?
    解決該問題的辦法是,恩,很笨,但是這可以工作:為不同參數個數的函數都實現一個對應的adapter。沒有參數的函數對
應一個adapter,一個參數的函數對應另一個adapter,依次類推。
    (穿插一下:ttl(tiny template library)庫中使用了一個很強大的宏技術,可以自動生成這些代碼,但是具體原理
我不懂。所以只能使用這個笨辦法了。)
    這樣,我們就需要區分不同參數個數的函數原型。很顯然,我們需要改進lua_binder。行之有效的技術是:模板偏特化。
改進后的lua_binder類似于:

    template <typename Prototype>
   
class lua_binder;

    template
<typename R, typename P1>

   
class lua_binder<R ( P1 )>
   
{
   
public
:
        typedef R result_type;
        typedef P1 p1_type;
        typedef result_type (
*
func_type)( P1 );
   
public
:
       
static int adapter( lua_State *
L )
       
{
           
return 0
;
        }

   
public:
       
static
func_type _func;
    }
;
    template
<typename R, typename P1>
 
    typename lua_binder
<R( P1 )>::func_type lua_binder<R( P1 )>::_func = 0
;

   
//



    lua_binder主體已經是一個單純的聲明而已,它的諸多特化版本將分別對應0個參數,1個參數,2個參數等。例如以上
列舉的就是一個參數的偏特化版本。

Now, we can try ??

    那么,我們現在是否可以寫下諸多的lua_to*函數去獲取參數了?你覺得可以嗎?假設現在要獲取棧頂第一個參數,你
該調用lua_tonumber還是lua_tostring?
    問題就在于,我們并不知道該調用哪個函數。
    解決辦法是:根據不同的參數類型,調用對應的lua_to*函數。
    不同的類型擁有不同的行為,這一點讓你想起什么?那就是模板世界里的type traits,類型萃取。我想,完成本文的
綁定器,更多的是對你模板編程能力的考驗。
    lua_to*系列函數是有限的,因此我們也只需要實現幾個類型的行為即可。我們這個時候的目的就是,根據不同的類型,
調用對應的lua_to*函數。例如,對于number(int, long, double, float, char等等),我們就調用lua_tonumber。
    于是得到:

    template <typename _Tp>
   
struct param_traits
   
{
       
static _Tp get_param( lua_State *L, int
index )
       
{
           
return static_cast<_Tp>
( lua_tonumber( L, index ) );
        }

    }
;

    template
<>

   
struct param_traits<const char*>
   
{
       
static const char *get_param( lua_State *L, int
index )
       
{
           
return
lua_tostring( L, index );
        }

    }
;
   
//others



    param_traits主體處理所有的number(因為number類型太多,也許concept可以解決這個問題),其他特化版本處理其他
類型。這樣,在adapter里,就可以根據參數類型獲取到相應的參數了,例如:

    P1 p1 = param_traits<P1>::get_param( L, -1);
    到這個時候,我們的adapter函數變為:(以一個參數的函數舉例)

    static int adapter( lua_State *L )
   
{
        p1_type p1
= param_traits<p1_type>::get_param( L, -1
);
        _func( p1 );

       
return 0
;
    }
 


And how about the result ??

    是的,我們還需要處理函數返回值。我們暫時假設所有的函數都只有一個返回值。這里面對的問題同取參數一樣,我
們需要根據不同的返回值類型,調用對應的lua_push*函數壓入返回值。
    同樣的type traits技術,你應該自己寫得出來,例如:

    template <typename _Tp>
   
struct return_traits
   
{
       
static void set_result( lua_State *
L, lua_Number r )
       
{
            lua_pushnumber( L, r );               
        }

    }
;
    template
<>

   
struct return_traits<const char*>
   
{
       
static void set_result( lua_State *L, const char *
r )
       
{
            lua_pushstring( L, r );
        }

    }
;


    到這個時候,我們的adapter函數基本成型了:

    static int adapter( lua_State *L )
   
{
        p1_type p1
= param_traits<p1_type>::get_param( L, -1
);
        result_type r
=
_func( p1 );
        return_traits
<result_type>
::set_result( L, r );
       
return 0
;
    }
 


The last 'return' ???

    最礙眼的,是adapter最后一行的return。LUA手冊上告訴我們,lua_CFunction必須返回函數返回值的個數。我們已經
假設我們只支持一個返回值,那么,很好,直接返回1吧。
    關鍵在于,C/C++的世界里還有個關鍵字:void。是的,它表示沒有返回值。在用戶層函數返回值為void類型時(原諒
這矛盾的說法),我們這里需要返回0。
    你意識到了什么?是的,我們需要根據返回值類型是否是void來設置這個return的值:1或者0。又是個type traits的
小技術。我想你現在很熟悉了:

    template <typename _Tp>
   
struct return_number_traits
   
{
       
enum

       
{
            count
= 1

        }
;
    }
;
    template
<>

   
struct return_number_traits<void>
   
{
       
enum

       
{
            count
= 0

        }
;
    }
;


    于是,我們的adapter變為:

    static int adapter( lua_State *L )
   
{
        p1_type p1
= param_traits<p1_type>::get_param( L, -1
);
        result_type r
=
_func( p1 );
        return_traits
<result_type>
::set_result( L, r );
       
return return_number_traits<result_type>
::count;
    }





Is everything OK?

    我很高興我能流暢地寫到這里,同樣我希望我不僅向你展示了某個應用的實現,而是展示了模板編程的思想。
    但是,問題在于,當你要注冊一個返回值為void的函數時:

    typedef lua_binder<void(int)> binder_type;
    lua_bind
<binder_type>( L, my_fn, "fn"
);


    你可能會被編譯器告知:非法使用void類型。
    是的,好好省視下你的代碼,當你的binder_type::result_type為void時,在adapter函數中,你基本上也就寫下了
void r = something 的代碼。這是個語法錯誤。

    同樣的問題還有:當返回值是void時,我們也沒有必要調用return_traits的set_result函數。
    我想你覺察出來,又一個type traits技術。我們將根據result_type決定不同的處理方式。于是,我寫了一個caller:

    template <typename _Tp>
   
struct caller
   
{
       
static void call( lua_State *L, p1_type &
p1 )
       
{
            result_type r
=
_func( p1 );
            return_traits
<result_type>
::set_result( L, r );
        }

    }
;

    template
<>

   
struct caller<void>
   
{
       
static void call( lua_State *L, p1_type &
p1 )
       
{
            _func( p1 );
        }

    }
;


    caller將根據不同的返回值類型決定如何去回調_func。比較遺憾的是,我們需要為每一個lua_binder編寫這么一個
caller,因為caller要調用_func,并且_func的參數個數不同。
    那么現在,我們的adapter函數變為:

    static int adapter( lua_State *L )
   
{
        P1 p1
= lua::param_traits<P1>::get_param( L, -1
);
        caller
<result_type>
::call( L, p1 );
       
return return_number_traits<result_type>
::count;
    }


END

    最后一段的標題不帶問號,所以這就結束了。下載看看我的代碼吧,為了給不同參數個數的函數寫binder,始終需要
粘貼復制的手工勞動。
    值得注意的是,在最終的代碼里,我使用了
以前實現的functor,將函數類型泛化。這樣,lua_binder就可以綁定類
成員函數,當然,還有operator()。

    開源代碼某個時候還需要勇氣,因為開源意味著你的代碼會被人們考驗。不過我對我代碼的風格比較自信。:D

    下載完整的lua_binder

posted on 2008-08-13 09:33 Kevin Lynx 閱讀(1040) 評論(5)  編輯 收藏 引用 所屬分類: C++Lua

評論

# re: 實現自己的LUA綁定器-一個模板編程挑戰 2008-08-13 09:47 dophi

kevin哥哥真厲害啊~  回復  更多評論  

# re: 實現自己的LUA綁定器-一個模板編程挑戰 2008-08-13 22:45 陳梓瀚(vczh)

根據經驗,開源之后,很少人會真的給你反饋什么。特別是在這里。  回復  更多評論  

# re: 實現自己的LUA綁定器-一個模板編程挑戰 2008-08-14 15:29 白云哥

模板用的很精彩,學習  回復  更多評論  

# re: 實現自己的LUA綁定器-一個模板編程挑戰[未登錄] 2008-08-14 22:52 清風徐來

nice work  回復  更多評論  

# re: 實現自己的LUA綁定器-一個模板編程挑戰[未登錄] 2008-08-17 23:04 Gohan

從你的文章中了解到了類型萃取,又是一個不小的收獲。  回復  更多評論  

posted on 2008-08-31 21:38 肥仔 閱讀(725) 評論(0)  編輯 收藏 引用 所屬分類: C++ 模板

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲精品免费网站| 亚洲乱码国产乱码精品精98午夜 | 亚洲激情一区| 久久在线免费观看视频| 亚洲激情精品| 亚洲欧洲一区| 亚洲一二区在线| 欧美日韩精品三区| 亚洲午夜一区二区| 久久本道综合色狠狠五月| 亚洲欧美国产77777| 国产欧美视频一区二区| 亚洲一级二级| 国产日韩欧美电影在线观看| 久久国产日韩| 激情校园亚洲| 欧美日韩国产综合网| 免费看亚洲片| 久久久久久九九九九| 欧美视频中文字幕在线| 久久久天天操| 亚洲色图自拍| 一区二区三区自拍| 亚洲午夜视频在线观看| 国内成+人亚洲+欧美+综合在线| 免费人成网站在线观看欧美高清| 一区二区欧美国产| 亚洲高清久久| 亚洲高清一区二| 国产精品久久久久久久7电影 | 亚洲永久免费精品| 久久午夜精品| 99riav国产精品| 一本久久a久久免费精品不卡| 亚洲在线播放电影| 亚洲国产精品久久久| 国产精品女主播在线观看| 久久婷婷一区| 国产欧美一区二区三区国产幕精品| 亚洲欧洲三级电影| 在线欧美一区| 99在线精品视频在线观看| 亚洲高清在线观看| 欧美视频在线观看一区| 欧美精品免费看| 美女日韩欧美| 免费精品99久久国产综合精品| 麻豆精品传媒视频| 日韩视频免费观看| 一区二区三区波多野结衣在线观看| 国产主播一区二区三区四区| 欧美国产日本韩| 欧美日韩中文另类| 欧美成人中文| 国产精品国产福利国产秒拍| 美女精品在线| 亚洲国产91| 一区二区三区视频在线看| 欧美一级在线视频| 久久综合九色99| 亚洲国产天堂久久国产91| 欧美一区二区三区视频在线| 日韩视频免费大全中文字幕| 亚洲最新视频在线| 欧美极品在线播放| 中文日韩在线| 亚洲第一区色| 久久裸体艺术| 欧美专区在线播放| 亚洲欧美日韩天堂| 免费短视频成人日韩| 亚洲日本va在线观看| 免费一级欧美片在线播放| 欧美亚洲自偷自偷| 午夜视频一区| 狠狠色狠色综合曰曰| 久久精品亚洲一区| 欧美黑人多人双交| 正在播放亚洲| 欧美午夜a级限制福利片| 久久久精品日韩| 国产精品欧美一区二区三区奶水| 欧美一区二区三区四区夜夜大片| 久久精品国产久精国产思思| 亚洲人成啪啪网站| 亚洲精品欧美精品| 欧美午夜视频网站| 久久久久久久综合日本| 亚洲高清不卡av| 亚洲国产精品日韩| 欧美主播一区二区三区美女 久久精品人 | 亚洲一级二级| 91久久精品日日躁夜夜躁国产| 国产精品最新自拍| 欧美成人免费视频| 亚洲午夜激情免费视频| 欧美精品在线视频| 欧美护士18xxxxhd| 久久嫩草精品久久久精品| 欧美在线免费观看视频| 久久亚洲免费| 农夫在线精品视频免费观看| 久久综合国产精品| 欧美成人午夜免费视在线看片| 国产午夜精品理论片a级探花 | 国产综合色一区二区三区| 亚洲国产一二三| 午夜免费久久久久| 亚洲欧美日韩成人| 西瓜成人精品人成网站| 一区二区视频免费完整版观看| 亚洲第一黄色| 国产精品99久久不卡二区| 在线观看日韩av电影| 99精品国产热久久91蜜凸| 99国内精品| 亚洲裸体俱乐部裸体舞表演av| 国产欧美日韩中文字幕在线| 国产一区欧美日韩| 久久婷婷综合激情| 欧美成人按摩| 欧美一级专区免费大片| 一本在线高清不卡dvd| 午夜精品久久久久久久99黑人| 亚洲第一综合天堂另类专| 亚洲天堂av图片| 欧美 亚欧 日韩视频在线| 黄色小说综合网站| 国产精品成人一区二区三区吃奶| 亚洲一区欧美一区| 国产在线不卡精品| 国内精品伊人久久久久av一坑| 99视频精品全国免费| 欧美经典一区二区三区| 国产一区二区三区免费不卡 | 欧美96在线丨欧| 亚洲网站啪啪| 国产精品视频精品视频| 亚洲精品欧洲精品| 久久久美女艺术照精彩视频福利播放 | 欧美一区二区三区成人| 狠狠色狠狠色综合日日tαg| 欧美日韩播放| 91久久精品国产| 在线视频日本亚洲性| 久久人人爽人人爽爽久久| 另类天堂av| 蜜臀av性久久久久蜜臀aⅴ| 久久亚洲一区二区| 亚洲网站啪啪| 欧美日韩三级| 亚洲精品乱码久久久久久蜜桃91| 欧美成在线观看| 久久国产精品99国产精| 亚洲资源av| 一区二区三欧美| 亚洲精品黄网在线观看| 欧美高清视频www夜色资源网| 亚洲免费大片| 欧美一级理论性理论a| 亚洲一区二区三区中文字幕在线| 亚洲福利视频在线| 久久婷婷综合激情| 欧美黄色日本| 国产美女一区二区| 久久激情视频久久| 亚洲国产精品久久久久秋霞影院| 久久久久成人网| 蜜乳av另类精品一区二区| 亚洲国产aⅴ天堂久久| 一本久久a久久免费精品不卡| 精品9999| 亚洲第一精品在线| 伊人狠狠色j香婷婷综合| 亚洲一区高清| 亚洲国产一区二区a毛片| 午夜天堂精品久久久久| 亚洲中无吗在线| 欧美日韩国产影院| 日韩系列在线| 亚洲茄子视频| 亚洲综合99| 亚洲欧美偷拍卡通变态| 久久久久久久一区二区三区| 国产精品乱人伦一区二区| 久久国产福利| 国产精品一区一区| 亚洲男人的天堂在线| 亚洲精品在线免费| 欧美成人国产| 亚洲国产小视频在线观看| 亚洲精品美女在线观看| 欧美在线精品免播放器视频| 亚洲国产综合在线| 影音先锋欧美精品| 久久三级福利| 久久精品视频在线看| 久久婷婷综合激情| 久久激情综合网| 久久久久免费视频|