/** 模板方法
*
* 用于创建不同的 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)
....;
}
}