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

cpp小菜園

簡單即是複雜

  C++博客 :: 首頁 :: 聯(lián)系 :: 聚合  :: 管理
  2 Posts :: 0 Stories :: 0 Comments :: 0 Trackbacks

常用鏈接

留言簿(3)

我參與的團隊

搜索

  •  

最新評論

閱讀排行榜

評論排行榜


  最近(其實也有一段時間了),在c++項目中需要用到類似java的反射機制,時間也比較趕,只能周圍看看有沒有現(xiàn)成的庫可用了,找到的資料也不算少,但多數(shù)都是關(guān)于動態(tài)創(chuàng)建類的,有點像工廠模式,而我需要的,是把實體類中的屬性反射出來。最后,找到了一個叫LibReflection的東西,雖然后來還是沒用上這個,但是稍微看了一下代碼,還是有不少收獲,所以就在這里記錄下來了,當(dāng)做學(xué)習(xí),也當(dāng)做對這個庫的一個分析吧。
  LibReflection是屬性和函數(shù)反射都有實現(xiàn)的,而整個庫竟然就一個h文件,本人精力有限,只把屬性反射抽取出來,相信離實際應(yīng)用也有距離,如果有人路過看到,請體諒別拍磚了。
  那么,首先看一下使用:

    TestClass inst;
    tat::tat_class *test_class = TestClass::get_class_ptr();
    std::map<std::string,tat::tat_field> field_map = test_class->get_fields();
    tat::tat_field test_vec_field = field_map.find("_vec_f")->second;
    std::vector<int> vec;
    test_vec_field.get(&inst,vec);
    vec.push_back(22);
    test_vec_field.set(&inst,vec);
    std::cout<<inst._vec_f[0]<<std::endl;

  TestClass類先不給出,反正就是該類有一個名字為“_vec_f”的屬性,類型是std::vector<int>,然后這段代碼通過反射修改_vec_f的內(nèi)容。。至于tat_class和tat_field,相信熟悉java的人都能猜出是什么了。
  獲取及設(shè)置一個實例的成員變量的值不是難事,用一個宏就可以了:

#define _OFFSET_(_Obj_Ty,_Key)                                                    \
    ((unsigned long)(&((_Obj_Ty *)0)->_Key))

   這個宏應(yīng)該也不難理解,就是獲取成員變量相對實例指針的偏移值。而關(guān)鍵問題是,如何通過變量名稱就能查找出這個偏移呢?如何在變量初始化的時候就能記錄下這個偏移呢?我們可以通過宏來聲明變量,那么聲明的時候就可以做一些事情了,但具體做些什么,還是沒頭緒。
  這個問題自己思考了一段時間,無果。然后看了一下實現(xiàn),深感自己的基礎(chǔ)還沒過關(guān),其實說白了就是兩個東西:構(gòu)造函數(shù)、靜態(tài)成員變量。
  我們的目標(biāo)是很明確的,要做到j(luò)ava的效果,結(jié)合上面那個宏,也不難想象出Class和Field類的樣子(這里是tat_class和tat_field),tat_class無非就是保存了一個map,key是屬性的名稱,value是tat_field,至于tat_field就簡單了,核心就是保存內(nèi)存的偏移值,再提供操作值的接口就可以了。
  先來看看tat_class:

    class tat_class
    {
    private:
        std::map<std::string,tat_field> _field_map;
        std::string _key;
    public:
        std::map<std::string,tat_field> get_fields()
        {
            return this->_field_map;
        }
        tat_field get_field(std::string key)
        {
            std::map<std::string,tat_field>::iterator itr = _field_map.find(key);
            return (*itr).second;
        }
        void add_field(const tat_field &field)
        {
            _field_map.insert(std::pair<std::string,tat_field>(field.get_key(),field));
        }
    };

  沒有什么理解上的難點吧。
  然后看看tat_field:

    class tat_field
    {
    private:
        unsigned long _offset;
        std::string _key;
    public:
        tat_field(unsigned long offset,std::string key):_offset(offset),_key(key){}
        tat_field(const tat_field &field)
        {
            this->_offset = field._offset;
            this->_key = field._key;
        }
    public:
        template<typename _Obj_Ty,typename _Value_Ty>
        void get(_Obj_Ty *obj,_Value_Ty &value)
        {
            value = *((_Value_Ty *)((unsigned char *)obj + _offset));
        }
        template<typename _Obj_Ty,typename _Value_Ty>
        void set(_Obj_Ty *obj,const _Value_Ty &value)
        {
            *((_Value_Ty *)((unsigned char *)obj + _offset)) = value;
        }
        std::string get_key() const
        {
            return this->_key;
        }
    };

  變量值的get和set函數(shù)使用了模板,這是為了使用的時候能作簡單的類型推斷。
  然后就是重點了,如何在聲明類的時候就把tat_class注入呢?如前面說的,靜態(tài)成員變量:

#define CLASS_REGISTER(_Obj_Ty)                                                    \
public:                                                                            \
    static tat::tat_class * get_class_ptr()                                        \
    {                                                                            \
        static tat::tat_class __class_##_Obj_Key##__;                            \
        return &__class_##_Obj_Key##__;                                            \
    }

  這個宏做了兩件事:第一,聲明靜態(tài)函數(shù)get_class_ptr(),返回tat_class類型,第二,當(dāng)然就是初始化tat_class了,因為同樣是靜態(tài),所以不同的實例就共享了。
  接下來是另一個重點,如何在聲明成員變量的時候把tat_field注入到tat_class中,還是用宏解決:

#define FIELD_REGISTER(_Access,_Field_Ty,_Field_Key,_Obj_Ty)                    \
_Access:                                                                        \
    _Field_Ty _Field_Key;                                                        \
private:                                                                        \
    class __field_register_##_Field_Key##__                                        \
    {                                                                            \
    public:                                                                        \
        __field_register_##_Field_Key##__()                                        \
        {                                                                        \
            static tat::__field_register__ reg_##_Field_Key(                    \
                _Obj_Ty::get_class_ptr(),                                        \
                _OFFSET_(_Obj_Ty,_Field_Key),                                    \
                #_Field_Key);                                                    \
        }                                                                        \
    }_Field_Key##_register;

  分析這個宏,首先當(dāng)然是聲明變量了;然后就是一個神秘的類:__field_register_##_Field_Key##__,這個類就只有一個構(gòu)造函數(shù),構(gòu)造函數(shù)里面又是一個靜態(tài)變量,類型是tat::__field_register__,可以想象得到,既然tat_class已經(jīng)可以聲明出來了,tat_field自然也沒有難度,但聲明是不足夠的,還需要放到tat_class的map里面才算完成,在聲明一個變量的時候,能做事的地方,我只能想到是構(gòu)造函數(shù)了,所以就有了剛剛那個神秘的類,那個類的唯一作用就是其構(gòu)造函數(shù),而構(gòu)造函數(shù)里面聲明靜態(tài)變量,則是防止多次實例化類帶來的冗余數(shù)據(jù),確保一個成員變量只有一個tat_field;
  剩下的關(guān)鍵,就是tat::__field_register__了:

    class __field_register__
    {
    public:
        __field_register__(tat_class *class_ptr,unsigned long offset,std::string key)
        {
            tat_field field(offset,key);
            class_ptr->add_field(field);
        }
    };

  沒有什么神秘的地方,這個東西就這么完成了。
  差點忘了TestClass的聲明:

#include "fieldref.h"

class TestClass
{
public:
    TestClass(void);
    ~TestClass(void);

    CLASS_REGISTER(TestClass)
    FIELD_REGISTER(public,long,_long_f,TestClass)
    FIELD_REGISTER(public,int,_int_f,TestClass)
    FIELD_REGISTER(public,std::string,_str_f,TestClass)
    FIELD_REGISTER(public,std::vector<int>,_vec_f,TestClass)
};

  好吧,如果說到實際應(yīng)用的問題,灑家確實沒有考慮太多,反正測試函數(shù)能跑,結(jié)果也對了,灑家也就滿足了,就單純的當(dāng)做是一次實驗,或者是學(xué)習(xí)而已。
posted on 2012-04-25 13:18 閱讀(4073) 評論(0)  編輯 收藏 引用

只有注冊用戶登錄后才能發(fā)表評論。
網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            99热在线精品观看| 亚洲制服少妇| 亚洲区中文字幕| 亚洲四色影视在线观看| 久久久久久网址| 国产精品日本一区二区| 日韩一级片网址| 国产综合一区二区| 亚洲女同在线| 亚洲精品一级| 欧美理论在线播放| 日韩视频在线观看一区二区| 麻豆乱码国产一区二区三区| 欧美一级二区| 国产亚洲精品一区二555| 欧美一区二区高清| 亚洲一区亚洲二区| 国产精品爽爽爽| 性久久久久久久久| 亚洲欧美高清| 国产日韩视频一区二区三区| 欧美一区免费视频| 午夜影视日本亚洲欧洲精品| 国产午夜精品理论片a级探花| 欧美在线亚洲在线| 久久精品国产精品亚洲精品| 激情欧美一区二区三区| 免费不卡在线视频| 欧美电影在线观看| 国产精品99久久久久久久vr| 亚洲色图在线视频| 国产一区91精品张津瑜| 久久人人爽国产| 狼人天天伊人久久| 在线一区二区三区四区| 亚洲欧美成人综合| 亚洲丰满少妇videoshd| 亚洲人成在线观看网站高清| 国产精品爱久久久久久久| 欧美一区二区三区视频在线| 欧美伊人影院| 99国产麻豆精品| 亚洲欧美日韩在线| 亚洲欧洲午夜| 亚洲精品国产精品乱码不99| 亚洲欧美日韩精品久久久| 亚洲欧美日产图| 在线观看欧美亚洲| 亚洲美女精品成人在线视频| 国产欧美日韩亚洲一区二区三区| 久久永久免费| 欧美日韩亚洲另类| 一区二区三区日韩| 欧美夜福利tv在线| 国产精品一级| 久久综合亚洲社区| 欧美日韩精品一区二区| 午夜精品福利电影| 六月婷婷久久| 欧美伊人久久| 欧美成人中文字幕在线| 久久成人一区| 欧美日韩成人在线| 免费观看国产成人| 国产精品日韩欧美综合 | 久久久久国色av免费观看性色| 久久在线视频在线| 欧美一级黄色录像| 欧美激情自拍| 裸体丰满少妇做受久久99精品| 欧美日韩大片一区二区三区| 美女福利精品视频| 国产麻豆成人精品| 亚洲人成亚洲人成在线观看图片| 国产亚洲一本大道中文在线| 一本色道久久加勒比88综合| 亚洲七七久久综合桃花剧情介绍| 欧美在线观看视频| 欧美影院视频| 国产精品亚洲综合| 99成人在线| 亚洲精品一区二区三区福利| 久久久久在线观看| 久久久久国产精品一区三寸| 国产精品嫩草影院av蜜臀| 亚洲视频观看| 欧美激情无毛| 亚洲高清网站| 亚洲国产精品久久久久秋霞蜜臀| 久久国产精品电影| 久久久久99| 精品av久久707| 久久久999国产| 六月天综合网| 一区二区三区在线看| 欧美专区福利在线| 久久久久久夜| 狠狠色狠狠色综合日日五| 欧美一区二区三区视频免费播放| 久久精品导航| 在线播放中文一区| 欧美国产国产综合| 亚洲美女视频在线观看| 亚洲神马久久| 国产精品视区| 久久精品国产91精品亚洲| 麻豆精品一区二区综合av| 在线观看成人一级片| 欧美成va人片在线观看| 日韩视频久久| 久久成人国产| 在线免费观看视频一区| 欧美二区不卡| 一区二区三区蜜桃网| 亚洲宅男天堂在线观看无病毒| 国产精品av免费在线观看| 亚洲欧美成人在线| 老司机aⅴ在线精品导航| 亚洲丶国产丶欧美一区二区三区| 欧美高清视频一区二区| 亚洲专区欧美专区| 久久婷婷丁香| 国产尤物精品| 免费视频一区二区三区在线观看| 亚洲国产精品女人久久久| 亚洲图片欧美一区| 国产日韩在线播放| 久久亚洲一区二区| 日韩一本二本av| 久久精品国产视频| 亚洲欧洲日韩在线| 国产精品欧美激情| 美女视频一区免费观看| 亚洲天堂av电影| 欧美成人国产| 午夜精品久久久久久久99黑人| 韩国精品在线观看| 国产精品二区三区四区| 久久亚洲国产精品日日av夜夜| 日韩午夜在线播放| 久久久久久999| 亚洲午夜视频在线| 最近看过的日韩成人| 国产伦精品一区二区三区| 欧美福利网址| 久久精品国产精品 | 免费成人小视频| 亚洲欧美日本国产专区一区| 亚洲免费观看高清完整版在线观看熊| 国产伦精品一区二区三区四区免费| 麻豆成人在线观看| 欧美在线观看视频| 亚洲一区日韩在线| 日韩视频免费观看高清在线视频 | 欧美在线播放一区| 夜夜嗨一区二区| 亚洲第一精品福利| 久久夜色精品国产噜噜av| 午夜久久福利| 亚洲一区美女视频在线观看免费| 在线免费观看日本一区| 国产丝袜一区二区三区| 国产精品九九久久久久久久| 欧美精品高清视频| 欧美大片一区二区三区| 麻豆成人精品| 快播亚洲色图| 久久久99国产精品免费| 亚洲一区欧美二区| 亚洲一区精品视频| 亚洲一区二区在线观看视频| 99re6热只有精品免费观看 | 国产精品天天看| 国产精品白丝av嫩草影院| 欧美区日韩区| 欧美日韩国产综合新一区| 欧美成人精品影院| 男女av一区三区二区色多| 免费观看成人| 欧美v日韩v国产v| 欧美成年人在线观看| 女女同性精品视频| 欧美精品一区二区三区很污很色的| 美日韩精品视频| 欧美sm视频| 欧美片在线播放| 欧美视频中文在线看| 国产精品久久久久999| 国产精品一区二区三区久久久| 国产欧美精品一区| 黄色影院成人| 亚洲精品欧美日韩专区| 夜夜爽www精品| 午夜精品婷婷| 先锋资源久久| 在线欧美日韩| 亚洲伦理久久| 亚洲综合激情| 久久久无码精品亚洲日韩按摩| 蜜桃av噜噜一区二区三区|