首页 > C++ 如何动态的创建对象

C++ 如何动态的创建对象

/** 模板方法
 *
 *  用于创建不同的 Card 实例
 */
template<typename T>
T* Create(int cardNum, char *name, char *password, int cardType)
{
    return new T(cardNum, name, password, cardType);
}


/** 发卡
 *
 * 传入参数:用户名
 *            密码
 *            卡的类型
 * 返回值:卡号
 */
int Bank::CreateCard(char *name, char *password, int cardType)
{
    int cardID;
    if(cardType == 1)  // 借记卡
    {
        //cards[CardNum] = new DebitCard(CardNum, name, password, cardType);
        cards[CardNum] = Create<CreditCard>(CardNum, name, password, cardType);
    }
    else if(cardType == 2)  // 贷记卡
    {
        //cards[CardNum] = new CreditCard(CardNum, name, password, cardType);
        cards[CardNum] = Create<DebitCard>(CardNum, name, password, cardType);
    }
    cardID = 1000 + CardNum;
    CardNum++;
    return cardID;
}

如上面的代码所示,起初我是想创建 Card 的子类对象时去除 if/else 语句的判断,所以定义了一个模板,但是调用模板时候仍然要传入具体的类型,在 CreatCard(...) 方法中,注释部分是之前的代码。如果我想去除 CreateCard(...) 方法中的 if/else 该怎么做呢?如果我用枚举值,C++ 中可以通过枚举字符串值转化为类型吗?


这个问题无卡反射,你需要使用工厂模式来负责创建对象。

如何来告诉工厂你想要创建的对象类型,这个方法有很多,简单列举几种:

1.使用枚举类型,自己记住每种枚举对应的对象类型,简单并且实用。工厂实用swith case来创建对象。或者使用各种你认为很cool的方法来产生对象。

CardFactory::CreateCard(enum CardTyppe)
{
    switch(CardType)
    {
       case enCreditCard:
            return new CreditCard();

       case enCreditCard:
            return newDebitCard();
        default:
            return nullptr;
    }
}

int Bank::CreateCard(...)
{
    CardFactory::CreateCard(cardTyppe);
}

2.C++中可以实用typeid来识别对象,注意这需要编译器支持,并且低效,并不推荐在这里使用。

使用工厂的好处在于,Bank不用关系对象创建的细节。只需要传递给工厂一个可以识别对象类型的标识,可以是枚举,也可以是字符串。


C++没有完善的反射机制,这点一直挺为人诟病。但是你这个需求很简单可以实现,思路供参考:

int create_credit_card(...) { ... };
int create_debit_card(...) { ... };

int Bank::CreateCard(...)
{
    struct { int type; int (*func)(...); } func_map[n] = {
        {1, create_credit_card},
        {2, create_debit_card},
        ...
    };

    for (i = 0; i < n; i++)
    {
        if (func_map[i].type == cardType)
            ....;
    }
}
【热门文章】
【热门文章】