Aliases: Abstract Factory, Kit
A, intentions
Provide an interface that is responsible for creating a set of “related or interdependent objects” without specifying their specific classes.
Second, motivation
In program development, we often face the creation of “a series of interdependent objects”. At the same time, as requirements change, there are often more series of objects to create.
The question is: what to do about it? How do I bypass the normal object creation method (New)?
The origin of factory model:
The usual way to create an object is:
CZYTest test = new CZYTest();
Copy the code
Problem with new: Implementation dependencies cannot handle “concrete instantiation type” changes.
Solution: encapsulate change point —- where change, encapsulate where. That is, if there is no change, no additional encapsulation is required.
Solution: Change point in “object creation”, so encapsulate “object creation”
Application Scenario: Abstract factory mode is mainly used to cope with the demand changes of “new series”, but difficult to cope with the demand changes of “new objects”.
Abstract factory patterns are often combined with factory method patterns to respond to changing requirements for “object creation”
Three, the structure is
- An Abstract Product is a set of different but related Product declaration interfaces that make up a family of products.
- Concrete products are different types of implementations of abstract products. All variants must implement the corresponding abstract product.
- The Abstract Factory interface declares a set of methods for creating various Abstract products.
- Concrete Factory implements the construction method of abstract Factory. Each specific factory corresponds to a specific product variant, and only that product variant is created.
Although a concrete factory initializes a concrete product, its build method signature must return the corresponding abstract product. In this way, client code that uses the factory class is not coupled to the specific product variant that the factory creates. Clients can interact with any concrete factory/product variant simply by invoking factory and product objects through an abstract interface.
Four, advantages and disadvantages
Advantages:
- Single responsibility principle: You can extract product-generated code into one location, making it easier to maintain.
- Open close rule: You don’t need to change the client code when introducing a new product variant into your application.
- Separation of concrete classes: The abstract factory pattern helps us control the classes in which an application creates objects. Because a factory encapsulates the responsibility and process of creating product pairs, it separates the implementation of the client program from the class. Clients manipulate instances through their abstract interfaces.
- Good for product consistency: When product objects in a family are designed to work together, it is important that an application only use objects in the same family at a time. The abstract factory pattern makes this easy.
Disadvantages:
- Code complexity: The code may be more complex than before because of the many interfaces and classes introduced into the application to adopt this pattern.
- Difficult to scale: difficult to scale to produce new kinds of products. This is because the abstract factory interface identifies the set of products that can be created. Supporting new kinds of products requires extending your factory interface, which will involve changes to the abstract factory class and all of its subclasses. (Large amount of code)
Five, application scenarios
Application Scenarios:
The abstract factory pattern can be used when:
- You can use an abstract factory if your code needs to interact with multiple related products from different families, but you don’t want to build your code based on a concrete class of the product because you don’t have access to the information in advance, or for future extensibility.
- Abstract Factories give you an interface to create objects for each family of products. As long as the code creates objects through this interface, you will not generate products that are inconsistent with the type of products already generated by the application.
- If you have a class that is based on a set of abstract methods, and its primary functionality is therefore unclear, you can consider using the abstract factory pattern in this case.
- In a good design program, each class does only one thing. If a class interacts with multiple types of products, consider extracting factory methods into a separate factory class or an abstract factory class with full functionality.
For example:
Take king Rongyao Xingyuan skin as an example:
A skin is basically made up of weapons, headdresses and bodies. Youth series: Youth weapons, youth headwear, youth body; Christmas series: Christmas weapons, Christmas headdress, Christmas body; Other series may be added later…
The objects that make up the skin are essentially constant, and the skin series is the point of change, so it is suitable for the abstract factory pattern.
Six, code implementation
Implementation method:
- Declare abstract product interfaces for all products. These interfaces (abstract classes) are then implemented by all concrete product classes.
- Declare an abstract factory interface (abstract class) and provide a set of build methods for all abstract products in the interface.
- Implement a concrete factory class for each product variant.
- Develop initialization code in your application. This code initializes a specific concrete factory class based on the application configuration or the current environment. The factory object is then passed to all classes that need to create the product.
- Find all direct calls to the product constructor in your code and replace them with calls to the corresponding build method in the factory object.
Sample code:
Again, take the above astral skin as an example:
- Create abstract factories – Skin factories, weapons, headgear, body factories.
// Weapon factory
public abstract class WuQiFactory{}// Headwear factory
public abstract class TouShiFactory{}// Body factory
public abstract class ShenTiFactory{}// Skin factory
public abstract class PiFuFactory
{
// Create weapon, headdress, body
public abstract WuQiFactory CreateWuQi();
public abstract TouShiFactory CreateTouShi();
public abstract ShenTiFactory CreateShenTi();
}
Copy the code
- Create concrete factory – Christmas collection
// The Christmas weapon is implemented
public class ShengDanWuQi : WuQiFactory{}// Christmas tiara
public class ShengDanTouShi : TouShiFactory{}// Christmas body
public class ShengDanShenTi : ShenTiFactory{}// Skin factory
public class ShengDanPiFuFactory : PiFuFactory
{
// Create weapon, headdress, body
public override WuQiFactory CreateWuQi()
{
Console.WriteLine("... Create the Christmas weapon...");
return new ShengDanWuQi();
}
public override TouShiFactory CreateTouShi()
{
Console.WriteLine("... Create a Christmas tiara...");
return new ShengDanTouShi();
}
public override ShenTiFactory CreateShenTi()
{
Console.WriteLine("... Create a Christmas body...");
return newShengDanShenTi(); }}Copy the code
Youth series – in the future, expand other series, directly according to this model can be created.
// The weapon of youth is realized
public class QinChunWuQi : WuQiFactory{}// Youth headdress
public class QinChunTouShi : TouShiFactory{}// Youth body
public class QinChunShenTi : ShenTiFactory{}// Skin factory
public class QinChunFuFactory : PiFuFactory
{
// Create weapon, headdress, body
public override WuQiFactory CreateWuQi()
{
Console.WriteLine("-- Create the weapon of youth --");
return new QinChunWuQi();
}
public override TouShiFactory CreateTouShi()
{
Console.WriteLine("-- Create youth headwear --");
return new QinChunTouShi();
}
public override ShenTiFactory CreateShenTi()
{
Console.WriteLine("-- Create a youthful body --");
return newQinChunShenTi(); }}Copy the code
- The mock client uses an abstract factory
// Simulate the client
class GameManager
{
PiFuFactory pifuFactory;
public GameManager(PiFuFactory pifuFactory)
{
this.pifuFactory = pifuFactory;
}
WuQiFactory wuQi;
TouShiFactory touShi;
ShenTiFactory shenTi;
// Create a skin
public void CreatePiFu()
{
Console.WriteLine("GameManager simulates the client to start creating the star element series skin..."); wuQi = pifuFactory.CreateWuQi(); touShi = pifuFactory.CreateTouShi(); shenTi = pifuFactory.CreateShenTi(); }}Copy the code
- C# Main method emulates a call:
class Program
{
static void Main(string[] args)
{
GameManager gm = new GameManager(new ShengDanPiFuFactory());
gm.CreatePiFu();
GameManager gm1 = new GameManager(newQinChunFuFactory()); gm.CreatePiFu(); Console.ReadKey(); }}Copy the code
The test results