Brief introduction:

In normal use, a client program would create a concrete implementation of an abstract factory, and then use the abstract factory as an interface to create concrete objects for the topic. The client program does not need to know (or care) about the specific types of objects it gets from these internal factory methods, because the client program only uses the generic interface to these objects. The abstract factory pattern separates the implementation details of a set of objects from their general use.

Contrast with factory methods:

The abstract factory pattern provides a way to encapsulate a set of separate factories with the same theme. A factory method is an implementation of a factory for a concrete object (product). Whereas abstract factory a factory corresponds to the creation of multiple concrete objects (products) of the same subject

Definition:

The essence of the abstract factory pattern is to “provide interfaces that create a series of related or independent objects without specifying the concrete classes of those objects.

UML:

Roles in the picture:

  • Abstract factory: a superclass of a concrete factory that provides an interface to create a series of different concrete objects (products)
  • Concrete factory: An object that implements a concrete object (product) used to determine the family or subject of the product
  • Abstract product: the superclass of the concrete product, which is exposed to the concrete object (product) type of the client
  • Concrete product: The actual type of concrete object, hiding the details of the implementation
  • Client: Code that uses the abstract factory interface

Example:

Suppose we have two product interfaces, Button and Border, each of which supports multiple families, such as the Mac family and the Windows family. In this way, the products of each series are MacButton, WinButton, MacBorder and WinBorder. To create a family of products at run time, we can create a factory MacFactory and WinFactory for each family of products. Each factory has two methods CreateButton and CreateBorder that return the corresponding product. These methods can be abstracted as an interface, AbstractFactory. This gives us the option to create the desired product family at run time

Struct MacButton:Button {} struct WinButton:Button {} protocol Border {}// Struct MacButton:Button {} struct WinButton:Button {} protocol Border { MacBorder:Border {} struct WinBorder:Border {} // Their factory protocol AbstractFactory {func CreateButton()->Button func CreateBorder()->Border } struct MacFactory: AbstractFactory{// MAC factory func CreateButton() -> Button {return MacButton()} func CreateBorder() -> Border {return MacBorder() } } struct WinFactory: AbstractFactory{//win factory func CreateButton() -> Button {return WinButton()} func CreateBorder() -> Border {return Var fac:AbstractFactory? Var fac:AbstractFactory? If style == 1 {fac = WinFactory()} else {fac = WinFactory()} let macbutton = fac! .CreateButton() //macbutton let macborad = fac! .CreateBorder() //macbuttonCopy the code

Models like variant factory methods are also common. Simple factory methods:

Struct MacButton:Button {} struct WinButton:Button {} protocol Border {}// Struct MacButton:Button {} struct WinButton:Button {} protocol Border {  MacBorder:Border {} struct WinBorder:Border {} enum Type { case Win case Mac } struct TypeFactory { static func createButton(type:Type)->Button{ let button:Button switch type { case Type.Win: button = WinButton() case Type.Mac: button = MacButton() } return button } static func createBorder(type:Type)->Border{ let board:Border switch type { case Type.Win: board = WinBorder() case Type.Mac: board = MacBorder() } return board } } let winBtn = TypeFactory.createButton(type: Typefactory.createborder (Type: type.mac)//MacBorCopy the code

Write like factory “method” instead of factory “class” :

//apple NSNumber let boolNum = NSNumber(booleanLiteral: true) let floatNum = NSNumber(floatLiteral: Class Button{static func creatMacBtn()->Button{return MacButton()} static func creatWinBtn()->Button{ return WinButton() } } class MacButton:Button {} class WinButton:Button {} let macBtn = Button.creatMacBtn()//macBtn let winBtn = Button.creatWinBtn()//winBtnCopy the code

Applicability:

  • The design of a set of related product objects needs to be emphasized for joint use
  • Provide a library of product classes and only want to show their interfaces, not their implementations

Advantages:

  • The specific product is separated from the customer code
  • Can change the product line

Disadvantages:

  • Extending new products in a product family is difficult and requires modifying the interface of the abstract factory

Reference:

  • Wikipedia – Abstract factory