Abstract Factory Design Pattern

Abstract factory pattern is categorized under creational design pattern that focuses on families of related object creation without exposing its internal logic.

In an abstract factory pattern, families of related objects are created without specifying their exact classes. It defines an interface for creating a set of related objects and allows different implementations of the factory to create different sets of related objects. 


Possible use cases for abstract factory pattern 

Here are some possible use cases for the Abstract Factory pattern

1. GUI Toolkits
Abstract Factory pattern is frequently used in GUI toolkits, where different UI components like buttons, text boxes, and menus need to be created for different platforms such as Windows, Mac, or Linux.

2. Database Drivers
Abstract Factory pattern can be used to create database drivers that support different databases like Oracle, MySQL, and SQL Server. The abstract factory can create a family of related objects like Connection, Statement, and ResultSet for each database.

3. Mobile App Development
Abstract Factory pattern can be used in mobile app development where different versions of the same app need to be created for different operating systems like iOS and Android. The abstract factory can create platform-specific UI components, APIs, and services.

4. Game Development
Abstract Factory patterns can be used in game development to create different types of objects like characters, weapons, and levels for different game genres like action, adventure, or strategy.

5. E-commerce Websites
Abstract Factory pattern can be used in e-commerce websites to create product catalogs for different categories like electronics, clothing, or books. The abstract factory can create a family of related objects like product images, descriptions, and reviews for each category.

Implementation

Implementation of the abstract factory pattern includes the following steps 

1. Create Interfaces for the productA and productB as shown 

  1. //Interface for the product A   
  2. public interface IProductA
  3. {
  4.     void ShowProductInfo();
  5. }

  6.  //Interface for the product B   
  7. public interface IProductB
  8. {
  9.     void ShowProductInfo();
  10. }


2. Create concrete classes for the interface as shown 

  1.  
  2.  //Concrete classes for product A  
  3. public class ProductA1 : IProductA
  4. {
  5.     public void ShowProductInfo()
  6.     {
  7.         Console.WriteLine("ProductA1 created");
  8.     }
  9. }

  10. public class ProductA2 : IProductA
  11. {
  12.     public void ShowProductInfo()
  13.     {
  14.         Console.WriteLine("ProductA2 created");
  15.     }
  16. }

  17.  //Concrete classes for product B
  18. public class ProductB1 : IProductB
  19. {
  20.     public void ShowProductInfo()
  21.     {
  22.         Console.WriteLine("ProductB1 created");
  23.     }

  24.  
  25. public class ProductB2 : IProductB
  26. {
  27.     public void ShowProductInfo()
  28.     {
  29.         Console.WriteLine("ProductB2 created");
  30.     }
 


3. Create an abstract factory with the help of an interface and classes which implements that interface and contains the logic of object creation as shown.

  1.  
  2. // Interface for the factory 
  3. public interface IProductFactory
  4. {
  5.     IProductA CreateProductA();
  6.     IProductB CreateProductB();
  7. }
  8. // Concrete classes for the factory 
  9. public class ProductFactory1 : IProductFactory
  10. {
  11.     public IProductA CreateProductA()
  12.     {
  13.         return new ProductA1();
  14.     }

  15.     public IProductB CreateProductB()
  16.     {
  17.         return new ProductB1();
  18.     }
  19. }


  20. public class ProductFactory2 : IProductFactory
  21. {
  22.     public IProductA CreateProductA()
  23.     {
  24.         return new ProductA2();
  25.     }

  26.     public IProductB CreateProductB()
  27.     {
  28.         return new ProductB2();
  29.     }
  30. }


4. Now create client class that uses the abstract factory as shown below.


  1.  //Client code that uses the abstract factory
  2. public class Client
  3. {
  4.     private IProductA _productA;
  5.     private IProductB _productB;

  6.     public Client(IProductFactory factory)
  7.     {
  8.         _productA = factory.CreateProductA();
  9.         _productB = factory.CreateProductB();

  10.     }

  11.     public void Run()
  12.     {
  13.         _productA.ShowProductInfo();
  14.         _productB.ShowProductInfo();
  15.     }
  16. }

5. You can now create different factories and use them to create the appropriate products:


  1. ProductFactory factory = new ProductFactory();
  2. IProductFactory factory1 = new ProductFactory1();

  3. Client client1 = new Client(factory1);

  4. //Creates ProductA1 and ProductB1
  5. client1.Run();

  6. IProductFactory factory2 = new ProductFactory2();

  7. Client client2 = new Client(factory2);

  8. //Creates ProductA2 and ProductB2
  9. client2.Run(); 


Advantages of Abstract Factory Pattern

1. Centralized related set of object creation logic that helps in code manageability.

2. Provides a level of abstraction for object creation.


Versioning in Abstract Factory Pattern

Handling versioning in an Abstract Factory implementation can be challenging, but there are several approaches that can be used depending on the specific requirements of the system. 

Here are some strategies that can be used

1. Versioning at the factory level

In this approach, each concrete factory is assigned a version number, and the abstract factory is modified to return the appropriate concrete factory based on the version number. This approach works well when there are only a few concrete factories, and the differences between versions are relatively minor.


2. Versioning at the product level

In this approach, each product returned by the abstract factory is assigned a version number, and the abstract factory is modified to return the appropriate version of the product based on the version number. This approach is more flexible than versioning at the factory level but can be more complex to implement.


3. Versioning through inheritance

In this approach, a new version of a product is created by subclassing the existing product, and the abstract factory is modified to return the appropriate subclass based on the version number. This approach works well when there are a large number of products, and the differences between versions are significant.


Here is an example of how versioning might be handled in an Abstract Factory implementation

Suppose you have an abstract factory called ReportFactory, which returns two products: Report and ReportTemplate. You want to introduce a new version of the Report product that includes additional data fields.

One approach to versioning at the product level could be to add a version parameter to the CreateReport method in the ReportFactory interface, like this

  1. // Interface for the factory 
  2. public interface IReportFactory 
  3. {
  4.     Report CreateReport(int version);
  5.     ReportTemplate CreateReportTemplate()
  6. }


The concrete factories implementing this interface would then check the version parameter and return the appropriate version of the Report product. For example:


  1. // Concrete classes for the factory for version 1 
  2. public class ReportFactoryV1 : IReportFactory 
  3. {
  4.     public Report CreateReport(int version)
  5.     {
  6.         return new ReportV1();
  7.     }

  8.     public ReportTemplate createReportTemplate()
  9.     {
  10.         return new ReportTemplateV1();
  11.     }
  12. }


  1. // Concrete classes for the factory for version 2 
  2. public class ReportFactoryV2 : IReportFactory 
  3. {
  4.     public Report CreateReport(int version)
  5.     { 
  6.        if(version == 2) 
  7.        {   
  8.         return new ReportV1();
  9.        }
  10.        else  
  11.        {   
  12.         return new ReportV1();
  13.        }
  14.     }

  15.     public ReportTemplate createReportTemplate()
  16.     {
  17.         return new ReportTemplateV2();
  18.     }
  19. }

In this example, the ReportV2 class would be a subclass of ReportV1, with additional data fields.

With this approach, clients of the ReportFactory interface can request a specific version of the Report product by passing in the version number when calling the createReport method.



Comments

Popular posts from this blog

Design Patterns

Azure Container Registry (ACR)

Factory Design Pattern

What is Azure DevOps?