如何像C++中的函数指针一样,但针对变量
how to do function pointer like in C++, but for variable
对不起标题,解释起来很困惑,但通过示例会更清楚
在C语言中,我曾经写过这样的程序:
void createClassA()
{
}
void createClassB()
{
}
typedef struct {
const char *name;
void (*create_class)();
} MapList;
MapList mapList[] = {
{"A", &createClassA},
{"B", &createClassB },
};
void create(const char *name)
{
int i=0;
for (i=0; i < sizeof(mapList) / sizeof(MapList); i++) {
if (strcmp(name, mapList[i].name) == 0) mapList[i].create_class();
}
}
int main(int argc, char** argv)
{
if (argc < 1) return -1;
create(argv[1]);
return 0;
}
因此,如果我有更多的类型,我所要做的就是创建函数本身并将其添加到mapList中。
但在C++中,我能做的最好的事情是:
#include <iostream>
class myA{};
class myB{};
class myC{};
class Base
{
public:
template<class T>
void createClass()
{
T* t = new T();
//t->doSomethingUseful();
}
void create(const std::string name)
{
if (name=="A") createClass<myA>();
else if (name=="B") createClass<myB>();
else if (name=="C") createClass<myC>();
}
};
int main(int agrc, char** argv)
{
if (argc<1) return -1;
Base b;
b.create(argv[0]);
return 0;
}
我可以创建这样的东西吗:
typedef struct {
std::string name;
[CLASS_TYPE] class_type; <== I don't know how
} MapList;
所以我可以创建一个映射列表,但用类类型填充它,类似
mapList[] = {
{"A", myA},
{"B", myB},
{"B", myC},
};
然后我可以创建迭代,比如:
for (int i=0; i<sizeof(mapList) / sizeof(MapList); i++) {
if (mapList[i].name == name) b.createClass<mapList[i].classType>();
}
谢谢大家:)
在C++中,您可以在<functional>
标准标头中使用std::function
。具体来说,您的地图列表示例可以写为(使用lambdas):
std::map<std::string, std::function<void()>> mapList;
mapList["A"] = []() { /* ... */ };
mapList["B"] = []() { /* ... */ };
或者只是(没有lambdas):
void createClassA() {}
void createClassB() {}
std::map<std::string, std::function<void()>> mapList;
mapList["A"] = createClassA;
mapList["B"] = createClassB;
在C++中,您可以使用虚拟函数:
struct ClassMaker {
virtual void create_class() = 0;
};
struct ClassMakerA : public ClassMaker {
virtual void create_class() {
...
}
};
struct ClassMakerB : public ClassMaker {
virtual void create_class() {
...
}
};
现在你可以用你的"工厂"创建一个std::map<std::string,ClassMaker*>
,如下所示:
ClassMakerA makerA;
ClassMakerB makerB;
// This syntax requires C++11
std::map<std::string,ClassMaker*> makers = {
{"A", &makerA}
, {"B", &makerB}
};
最后,您现在可以根据std::string
值调用create_class()
,如下所示:
void create(const std::string &name) {
ClassMaker *maker = makers[name];
if (maker) maker -> create_class();
}
您可以查找Boost::any
,它提供了"可变值类型"。
您可以使用派生类和对象工厂,这样就可以了。您可以通过使create方法返回智能指针,并使A和B构造函数私有以避免在堆栈上创建对象来改进它。
class Base
{
static Base* create(const char *name);
};
class A : public Base
{
};
class B : public Base
{
};
Base* Base::create(const char *name)
{
if(strcmp(name,"A")==0)
return new A();
if(strcmp(name,"B")==0)
return new B();
return 0;
}
int main(int agrc, char** argv)
{
if (argc<1) return -1;
Base *b = Base::create(argv[0]);
return 0;
}
它与@dasblinkenlight的解决方案有些相似,但避免了虚拟函数开销。
相关文章:
- 如何创建一个CMake变量,除非显式重写,否则使用默认值
- 将成员变量添加到共享库中的类中,不会破坏二进制兼容性吗
- 我们可以将集合的值存储在变量中吗?就像我们可以将数组的值存储在变量中一样
- 如何将两个字符串加在一起,就好像它们是变量一样?
- 变量模板,具有与实例一样多的参数
- 为什么向量不支持 size 作为成员变量,就像 Java 类中的长度变量一样,而是函数 size()?
- 'inline'变量可以像内联函数一样内联吗?
- 什么形式上保证了非原子变量不能看到凭空出现的值,并像理论上原子一样创造数据竞赛?
- 使用类型推断声明变量是否与"classical way"变量名称后用括号初始化变量一样有效?
- 模板成员变量像通用函数一样键入
- 我什么时候可以确定一个 constexpr 全局变量会像 C 宏一样被"forgotten"?
- 如何将实部和虚部放入一个变量中,就像在matlab中一样
- 如何像C++中的函数指针一样,但针对变量
- 在C++中,我可以像在Java中一样拥有一个接口变量吗
- 输入Std::cin与直接赋值Std::string变量不一样
- 需要学习如何通过指针正确访问实例变量"this"就像在 Java 中一样
- c++标准和C语言在哪里说的是一样的:编译单元(.cpp文件)中的变量是按照声明的顺序初始化的
- 监视器和条件变量,它们是一样的吗?
- 调用函数就像声明变量一样
- 为什么Java方法中不允许使用静态变量,就像在C/C++函数中一样