本文編寫 visualfc
寫一個invoke函數以支持任意參數調用,實現類似以下的功能代碼。注:本代碼純屬試驗。
- class a
- {
- void func1(int,int);
- void func2(char*,int);
- };
- void invoke(a*,char * method,...);
- void test()
- {
- a a1,
- invoke(&a1,"func1",10,20);
- invoke(&a2,"func2","ok",0);
- }
我目前試驗了三種方式,第一種是模板方式實現,第二種是基于可變參數(...)實現,第三種是使用類似boost::any的方式來實現。比較三種方式,
模板方式實現起來尤其不易,但是能夠做參數檢查,問題在于只能存在在頭文件中。 第二種為可變參數方式,
本文給出了這種方式的試驗代碼,實現起來也比較簡單,問題是無法驗證參數。第三種方式為以類似any的方式來實現,但是實現效率則相對較低。
下面給出了第二種方式,即使用可變參數調用的實現,沒有使用va_list和va_arg,這是因為函數調用棧順序的原因,我使用了自定義的get_arg_back函數來獲取可變參數。
- template <typename T>
- T get_arg_back(char ** parg)
- {
- *parg -= sizeof(T);
- T t = *((T*)*parg);
- return t;
- }
- template <typename R, typename CLS, typename T1, typename T2>
- R invoke_cls(CLS * pcls, R (CLS::*ptr)(T1,T2), char ** parg)
- {
- char * end = *parg+sizeof(T1)+sizeof(T2);
- return (pcls->*ptr)(get_arg_back<T1>(&end),get_arg_back<T2>(&end));
- }
- class a
- {
- public:
- int test(char * buf, int i)
- {
- printf ("test %s %d\n",buf,i);
- return 0;
- }
- int test2(int i, char * buf)
- {
- printf ("test2 %s %d\n",buf,i);
- return 100;
- }
- };
- void invoke(void * ptr, char * method,...)
- {
- char * parg = (char*)&method+sizeof(method);
- if (method == "test")
- invoke_cls((a*)ptr,&a::test,&parg);
- else if (method == "test2")
- invoke_cls((a*)ptr,&a::test2,&parg);
- parg = NULL;
- }
- int main(int argc, char* argv[])
- {
- a a1;
- invoke(&a1,"test","ok",1);
- invoke(&a1,"test2",1,"ok");
- return 0;
- }