C++ Builder 的反射 (三) - 通用 Reflection Factory

C++ Builder 参考手册C++ Builder 的反射 (三) - 通用 Reflection Factory


因为前面一篇文章 C++ Builder 获取任意一个类或对象的类名 已经找到了获取任意一个类或对象的类名的方法,只需要把 C++ Builder 的反射 (二) - Reflection Factory 的模板里面的 TClassType::ClassName() 改成 THsuanluClassName::GetClassName<TClassType>() 就可以了,这样无论是否从 TObject 继承的类都可以反射了。这个类模板需要 C++ 11 / clang 编译器。本文的程序和例子已经用 C++ Builder 10.2.3 版本 clang 32位 和 clang 64位编译器测试通过。

  1. 通用 Reflection Factory 类模板
  2. 测试控件的反射 - 控件都是 TObject 继承的类
  3. 测试自己写的类的反射 - 不是 TObject 继承的类

1. 通用 Reflection Factory 类模板

THsuanluReflectionFactory 类模板的参数:

  • TBaseClass 为可以反射的类的公共父类,通过类名字符串反射创建的类通过这个类型的指针返回;
  • TConstructorParams 为反射的类的构造函数的参数的类型,这不是单一的参数,而是一组随意个数和类型的参数,也可以没有。由于创建类的时候,必需调用类的构造函数,所以需要提供类的构造函数的每个参数的类型,请参考 C++ 可变参数的模板,构造函数参数相同的类,可以注册到这个 Reflection Factory 里面;

Register 方法为注册需要反射的类,模板参数 TClassType 为要注册的类;
Create 方法为通过类名字符串创建类,参数 sClassName 为类名字符串,参数 Params 为构造函数的可变参数,通过 new TClassType(Params...) 创建注册的类。

模板里面的 THsuanluClassName::GetClassName 是 C++ Builder 获取任意一个类或对象的类名 介绍的方法,即把 THsuanluClassName 类加入头文件,这个类的实现加入 cpp 文件。

#include <map>
template <class TBaseClass, class... TConstructorParams> // 公共父类, 构造函数参数类型
class THsuanluReflectionFactory
{
private:
    std::map<UnicodeString, TBaseClass*(*)(TConstructorParams...)>_ClassMap;

public:
    template<class TClassType> // 要注册的类
    void Register(void) // 注册 TClassType 类
    {
        _ClassMap[THsuanluClassName::GetClassName<TClassType>()] =
        [](TConstructorParams... Params) -> TBaseClass*
        {
            return new TClassType(Params...);
        };
    }

    TBaseClass *Create(UnicodeString sClassName, TConstructorParams... Params)
    {
        auto iter = _ClassMap.find(sClassName);
        if(iter == _ClassMap.end())
            throw Exception(L"类 \"" + sClassName + L"\" 未注册");
        return iter->second(Params...);
    }
};

2. 测试控件的反射 - 控件都是 TObject 继承的类

  • 这个例子在程序开始运行时,即在主窗口的构造函数里面,注册了 TLabel、TButton、TMemo、TEdit、TCheckBox、TRadioButton、TComboBox 等几个控件类,可以使用他们的类名字符串来创建控件对象。
  • 由于控件类的构造函数都有一个 TComponent* Owner 参数,在定义 ControlsFactory 的时候,模板参数除了公共父类 TControl 之外,还有一个 TComponent* 参数。
THsuanluReflectionFactory<TControl, TComponent*> ControlsFactory;
//---------------------------------------------------------------------------
__fastcall TForm1::TForm1(TComponent* Owner)
    : TForm(Owner)
{
    ControlsFactory.Register<TLabel>();
    ControlsFactory.Register<TButton>();
    ControlsFactory.Register<TMemo>();
    ControlsFactory.Register<TEdit>();
    ControlsFactory.Register<TCheckBox>();
    ControlsFactory.Register<TRadioButton>();
    ControlsFactory.Register<TComboBox>();
}
//---------------------------------------------------------------------------
void __fastcall TForm1::ButtonCreateClick(TObject *Sender)
{
    try
    {
        TControl *p = ControlsFactory.Create(EditClassName->Text, this);
        p->Parent = this;
        p->SetBounds(EditLeft->Text.ToIntDef(0), EditTop->Text.ToIntDef(0), EditWidth->Text.ToIntDef(80), EditHeight->Text.ToIntDef(30));
    }
    catch(Exception &e)
    {
        ShowMessage(e.Message);
    }
}

编辑框 EditClassName 输入控件的类名,
编辑框 EditLeft, EditTop, EditWidth, EditHeight 分别用于输入控件的位置和大小,
按钮 ButtonCreate 创建控件,ButtonCreateClick 方法就是点击这个按钮执行的代码。

测试使用 Reflection Factory 控件反射 - 窗口设计

运行结果:

创建一个 TComboBox:

创建一个 TComboBox

再创建一个 TCheckBox 和一个 TRadioButton

再创建一个 TCheckBox 和一个 TRadioButton

3. 测试自己写的类的反射 - 不是 TObject 继承的类

THsuanluBase 为其他几个类的公共父类,反射的结果是通过这个类型的指针返回通过类名字符串创建的对象。要注意:通过父类指针访问子类的方法要定义为虚函数。

class THsuanluBase
{
public:
    THsuanluBase(){}
    virtual ~THsuanluBase(){}
    virtual void PrintMessage(void){ Form1->Memo1->Lines->Add(L"这是父类的 PrintMessage 方法"); }
};
//---------------------------------------------------------------------------
class THsuanluTest1 : public THsuanluBase
{
public:
    virtual void PrintMessage(void){ Form1->Memo1->Lines->Add(L"这是 THsuanluTest1 的 PrintMessage 方法"); }
};
//---------------------------------------------------------------------------
class 玄坴测试类 : public THsuanluBase
{
public:
    virtual void PrintMessage(void){ Form1->Memo1->Lines->Add(L"这是 玄坴测试类 的 PrintMessage 方法"); }
};
//---------------------------------------------------------------------------
class THsuanluTest2 : public 玄坴测试类
{
public:
    virtual void PrintMessage(void){ Form1->Memo1->Lines->Add(L"这是 THsuanluTest2 的 PrintMessage 方法"); }
};
  • 定义 HsuanluFactory 时,由于自己写的这些类的构造函数没有参数,所以模板参数只有一个公共父类 THsuanluBase
  • 在程序开始执行的时候,即主窗口的构造函数里面注册所有需要反射的类,然后就可以使用类名字符串来创建这些类的对象了。
THsuanluReflectionFactory<THsuanluBase> HsuanluFactory;
//---------------------------------------------------------------------------
__fastcall TForm1::TForm1(TComponent* Owner)
    : TForm(Owner)
{
    HsuanluFactory.Register<THsuanluTest1>();
    HsuanluFactory.Register<玄坴测试类>();
    HsuanluFactory.Register<THsuanluTest2>();
}
//---------------------------------------------------------------------------
void __fastcall TForm1::Button1Click(TObject *Sender)
{
    THsuanluBase *p1 = nullptr, *p2 = nullptr, *p3 = nullptr;
    try
    {
        p1 = HsuanluFactory.Create(L"THsuanluTest1");
        p2 = HsuanluFactory.Create(L"玄坴测试类");
        p3 = HsuanluFactory.Create(L"THsuanluTest2");

        p1->PrintMessage();
        p2->PrintMessage();
        p3->PrintMessage();
    }
    catch(Exception &e)
    {
        ShowMessage(e.Message);
    }
    delete p1;
    delete p2;
    delete p3;
}

运行结果:

测试反射的运行结果

相关:


C++ Builder 参考手册C++ Builder 的反射 (三) - 通用 Reflection Factory

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,470评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,393评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,577评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,176评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,189评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,155评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,041评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,903评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,319评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,539评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,703评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,417评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,013评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,664评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,818评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,711评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,601评论 2 353

推荐阅读更多精彩内容