How to implement the factory method pattern in C++ correctly

后端 未结 11 1086
暗喜
暗喜 2020-11-22 14:55

There\'s this one thing in C++ which has been making me feel uncomfortable for quite a long time, because I honestly don\'t know how to do it, even though it sounds simple:<

相关标签:
11条回答
  • 2020-11-22 15:21

    You can read a very good solution in: http://www.codeproject.com/Articles/363338/Factory-Pattern-in-Cplusplus

    The best solution is on the "comments and discussions", see the "No need for static Create methods".

    From this idea, I've done a factory. Note that I'm using Qt, but you can change QMap and QString for std equivalents.

    #ifndef FACTORY_H
    #define FACTORY_H
    
    #include <QMap>
    #include <QString>
    
    template <typename T>
    class Factory
    {
    public:
        template <typename TDerived>
        void registerType(QString name)
        {
            static_assert(std::is_base_of<T, TDerived>::value, "Factory::registerType doesn't accept this type because doesn't derive from base class");
            _createFuncs[name] = &createFunc<TDerived>;
        }
    
        T* create(QString name) {
            typename QMap<QString,PCreateFunc>::const_iterator it = _createFuncs.find(name);
            if (it != _createFuncs.end()) {
                return it.value()();
            }
            return nullptr;
        }
    
    private:
        template <typename TDerived>
        static T* createFunc()
        {
            return new TDerived();
        }
    
        typedef T* (*PCreateFunc)();
        QMap<QString,PCreateFunc> _createFuncs;
    };
    
    #endif // FACTORY_H
    

    Sample usage:

    Factory<BaseClass> f;
    f.registerType<Descendant1>("Descendant1");
    f.registerType<Descendant2>("Descendant2");
    Descendant1* d1 = static_cast<Descendant1*>(f.create("Descendant1"));
    Descendant2* d2 = static_cast<Descendant2*>(f.create("Descendant2"));
    BaseClass *b1 = f.create("Descendant1");
    BaseClass *b2 = f.create("Descendant2");
    
    0 讨论(0)
  • 2020-11-22 15:21

    Loki has both a Factory Method and an Abstract Factory. Both are documented (extensively) in Modern C++ Design, by Andei Alexandrescu. The factory method is probably closer to what you seem to be after, though it's still a bit different (at least if memory serves, it requires you to register a type before the factory can create objects of that type).

    0 讨论(0)
  • 2020-11-22 15:24

    Simple Factory Example:

    // Factory returns object and ownership
    // Caller responsible for deletion.
    #include <memory>
    class FactoryReleaseOwnership{
      public:
        std::unique_ptr<Foo> createFooInSomeWay(){
          return std::unique_ptr<Foo>(new Foo(some, args));
        }
    };
    
    // Factory retains object ownership
    // Thus returning a reference.
    #include <boost/ptr_container/ptr_vector.hpp>
    class FactoryRetainOwnership{
      boost::ptr_vector<Foo>  myFoo;
      public:
        Foo& createFooInSomeWay(){
          // Must take care that factory last longer than all references.
          // Could make myFoo static so it last as long as the application.
          myFoo.push_back(new Foo(some, args));
          return myFoo.back();
        }
    };
    
    0 讨论(0)
  • 2020-11-22 15:27

    First of all, there are cases when object construction is a task complex enough to justify its extraction to another class.

    I believe this point is incorrect. The complexity doesn't really matter. The relevance is what does. If an object can be constructed in one step (not like in the builder pattern), the constructor is the right place to do it. If you really need another class to perform the job, then it should be a helper class that is used from the constructor anyway.

    Vec2(float x, float y);
    Vec2(float angle, float magnitude); // not a valid overload!
    

    There is an easy workaround for this:

    struct Cartesian {
      inline Cartesian(float x, float y): x(x), y(y) {}
      float x, y;
    };
    struct Polar {
      inline Polar(float angle, float magnitude): angle(angle), magnitude(magnitude) {}
      float angle, magnitude;
    };
    Vec2(const Cartesian &cartesian);
    Vec2(const Polar &polar);
    

    The only disadvantage is that it looks a bit verbose:

    Vec2 v2(Vec2::Cartesian(3.0f, 4.0f));
    

    But the good thing is that you can immediately see what coordinate type you're using, and at the same time you don't have to worry about copying. If you want copying, and it's expensive (as proven by profiling, of course), you may wish to use something like Qt's shared classes to avoid copying overhead.

    As for the allocation type, the main reason to use the factory pattern is usually polymorphism. Constructors can't be virtual, and even if they could, it wouldn't make much sense. When using static or stack allocation, you can't create objects in a polymorphic way because the compiler needs to know the exact size. So it works only with pointers and references. And returning a reference from a factory doesn't work too, because while an object technically can be deleted by reference, it could be rather confusing and bug-prone, see Is the practice of returning a C++ reference variable, evil? for example. So pointers are the only thing that's left, and that includes smart pointers too. In other words, factories are most useful when used with dynamic allocation, so you can do things like this:

    class Abstract {
      public:
        virtual void do() = 0;
    };
    
    class Factory {
      public:
        Abstract *create();
    };
    
    Factory f;
    Abstract *a = f.create();
    a->do();
    

    In other cases, factories just help to solve minor problems like those with overloads you have mentioned. It would be nice if it was possible to use them in a uniform way, but it doesn't hurt much that it is probably impossible.

    0 讨论(0)
  • 2020-11-22 15:28

    Have you thought about not using a factory at all, and instead making nice use of the type system? I can think of two different approaches which do this sort of thing:

    Option 1:

    struct linear {
        linear(float x, float y) : x_(x), y_(y){}
        float x_;
        float y_;
    };
    
    struct polar {
        polar(float angle, float magnitude) : angle_(angle),  magnitude_(magnitude) {}
        float angle_;
        float magnitude_;
    };
    
    
    struct Vec2 {
        explicit Vec2(const linear &l) { /* ... */ }
        explicit Vec2(const polar &p) { /* ... */ }
    };
    

    Which lets you write things like:

    Vec2 v(linear(1.0, 2.0));
    

    Option 2:

    you can use "tags" like the STL does with iterators and such. For example:

    struct linear_coord_tag linear_coord {}; // declare type and a global
    struct polar_coord_tag polar_coord {};
    
    struct Vec2 {
        Vec2(float x, float y, const linear_coord_tag &) { /* ... */ }
        Vec2(float angle, float magnitude, const polar_coord_tag &) { /* ... */ }
    };
    

    This second approach lets you write code which looks like this:

    Vec2 v(1.0, 2.0, linear_coord);
    

    which is also nice and expressive while allowing you to have unique prototypes for each constructor.

    0 讨论(0)
提交回复
热议问题