今天简单写了个Factory Method,结果没有效果,很是纳闷。研究之后发现还是自己c++基础不牢固阿!
#include <stdio.h>
class Factory
{
public:
virtual void createSomething() = 0;
};
class Child1Factory : public Factory
{
public:
virtual void createSomething() { printf("Child1Factory::%s()\n", __func__);
};
class Child2Factory : public Factory
{
public:
virtual void createSomething() { printf("Child2Factory::%s()\n", __func__);
};
class Parent
{
public:
Parent() { m_factory = create(); } // 在构造函数中调用虚函数来实现多态,想法是好的,但是实现不了,下面解释
private:
virtual Factory * create() { return new Child1Factory; } // 如果是纯虚函数,ld出错,下面解释
private:
Factory * m_factory;
};
class Child : public Parent
{
public:
Child() : Parent() {}
private:
virtual Factory * create() { return new Child2Factory; }
};
int main(void)
{
Child c;
c.factory()->createSomething();
}
输出结果是:Child1Factory::createSomething()
如果红色代码改为纯虚函数,链接的时候就会出错:undefined reference to 'Parent::create()'
在http://www.cnblogs.com/qytan36/archive/2010/12/02/1894123.html找到了答案。
解释一下原因:
1. 为什么父类定义为纯虚函数就会出现连接错误。
因为在父类构造函数中调用的函数指定属于父类,应为此时所有的子类都没有构造出来(子类构造之前总会调用到父类的构造函数,所以任何子类在父类构造函数之前都没有被构造出来)。所以此时调用父类的纯虚函数自然就会连接不到了(根本就没有实现)
2. 结果为什么是Child1Factory::createSomething()
有了1的解答,2的答案就非常明显了
引用以下“宁静的水泡”的文字:
1。在C++中,对于函数调用的多态主要是通过虚表实现,在构造函数完成前,虚表未实现,所以此时不会有多态特性。——即构造函数中没有多态特性
2。Java中表现则不是的,会调用到子类的方法。在Java中,若在构造时保证调用到的不会是派生类的方法,一定要调用private, 或final方法。因为final方法不会被覆盖,private也是,private默认其实就是final的,覆盖private方法其实是生成了一个新方法。