我正在尝试在加载时向工厂注册一堆类。我的策略是利用静态初始化来确保在 main() 开始之前,工厂已准备就绪。当我动态链接我的库时,这种策略似乎有效,但当我静态链接时无效;当我静态链接时,只有我的一些静态数据成员被初始化。
假设我的工厂生产汽车。我有 CarCreator 类可以实例化少量汽车,但不是全部。我希望工厂收集所有这些 CarCreator 类,以便寻找新车的代码可以进入工厂,而无需知道谁将进行实际构建。
所以我有
CarTypes.hpp
enum CarTypes
{
prius = 0,
miata,
hooptie,
n_car_types
};
我的工厂.hpp
class CarCreator
{
public:
virtual Car * create_a_car( CarType ) = 0;
virtual std::list< CarTypes > list_cars_I_create() = 0;
};
class MyFactory // makes cars
{
public:
Car * create_car( CarType type );
void factory_register( CarCreator * )
static MyFactory * get_instance(); // singleton
private:
MyFactory();
std::vector< CarCreator * > car_creator_map;
};
我的工厂.cpp
MyFactory:: MyFactory() : car_creator_map( n_car_types );
MyFactory * MyFactory::get_instance() {
static MyFactory * instance( 0 ); /// Safe singleton
if ( instance == 0 ) {
instance = new MyFactory;
}
return instance;
}
void MyFactory::factory_register( CarCreator * creator )
{
std::list< CarTypes > types = creator->list_cars_I_create();
for ( std::list< CarTypes >::const_iteator iter = types.begin();
iter != types.end(); ++iter ) {
car_creator_map[ *iter ] = creator;
}
}
Car * MyFactory::create_car( CarType type )
{
if ( car_creator_map[ type ] == 0 ) { // SERIOUS ERROR!
exit();
}
return car_creator_map[ type ]->create_a_car( type );
}
...
然后我会有特定的汽车和特定的汽车创造者:
Miata.cpp
class Miata : public Car {...};
class MiataCreator : public CarCreator {
public:
virtual Car * create_a_car( CarType );
virtual std::list< CarTypes > list_cars_I_create();
private:
static bool register_with_factory();
static bool registered;
};
bool MiataCreator::register_with_factory()
{
MyFactory::get_instance()->factory_register( new MiataCreator );
return true;
}
bool MiataCreator::registered( MiataCreator::register_with_factory() );
...
重申一下:动态链接我的库,MiataCreator::registered 将被初始化,静态链接我的库,它不会被初始化。
使用静态构建,当有人到工厂请求 Miata 时,miata 元素car_creator_map
将指向 NULL,程序将退出。
私有静态整数数据成员有什么特别之处,它们的初始化会以某种方式被跳过吗?是否仅在使用类时才初始化静态数据成员?我的 CarCreator 类没有在任何头文件中声明;它们完全存在于 .cpp 文件中。编译器是否有可能内联初始化函数并以某种方式避免调用 MyFactory:: factory_register
?
这个注册问题有更好的解决方案吗?
在单个函数中列出所有 CarCreators 并在工厂中显式注册每个 CarCreators 并确保调用该函数不是一种选择。特别是,我想将几个库链接在一起并在这些单独的库中定义 CarCreators,但仍然使用单个工厂来构造它们。
...
以下是我期待的一些回应,但不能解决我的问题:
1)你的单身工厂不是线程安全的。a) 没关系,我只使用一个线程。
2) 当您的 CarCreators 正在初始化时,您的单例工厂可能未初始化(即您有一个静态初始化失败) a) 我通过将单例实例放入函数中来使用单例类的安全版本。如果这是一个问题,如果我在方法中添加了一个打印语句,我应该会看到输出MiataCreator's::register_with_factory
:我没有。