Abstract Factory¶
What ?¶
The Abstract Factory Pattern is a creational design pattern that provides an interface for creating families of related or dependent objects without specifying their concrete classes. It promotes loose coupling between client code and the actual implementations, allowing the code to be more flexible and scalable.
The Abstract Factory pattern works as a super-factory that creates other factories. Each factory produced by the abstract factory is responsible for creating a family of related objects.
Key Characteristics
- Encapsulates the creation logic of families of related objects.
- Supports product families (like buttons and scrollbars for different OS themes).
- Makes the system open for extension but closed for modification (Open-Closed Principle).
- Clients interact with the factory interface instead of the concrete classes.
Class Diagram
When to Use ?¶
- When your application needs to create a family of related objects (e.g., GUI components for themes, or different database connections).
- When the code should be decoupled from the actual product classes.
- If you need to switch between different configurations (e.g., Light vs Dark theme).
Advantages¶
- Abstract factories make it easy to reuse code across product families.
- Supports the Open/Closed Principle you can introduce new product families without modifying existing code.
- Encourages Loose Coupling reduces dependencies between client code and the actual implementation of products.
- Easy Maintenance since product creation logic is centralized, maintenance is simpler.
Disadvantages¶
- The pattern introduces multiple classes and interfaces, which can make the design more complex.
- If there are too many products or product families, maintaining multiple factories may become cumbersome.
- For simple object creation tasks, other patterns like Factory Method or just using constructors may be more suitable.
How to Implement ?¶
Simple Example
Let’s go with an example Imagine you are creating a UI component factory. Your application can switch between two themes: Dark Theme and Light Theme. Both themes provide the same types of components (buttons, text fields) but with different appearances.
// Abstract Product: Button
public interface Button {
void render();
}
// Abstract Product: TextField
public interface TextField {
void render();
}
// Concrete Product: Light Button
public class LightButton implements Button {
@Override
public void render() {
System.out.println("Rendering a Light Button");
}
}
// Concrete Product: Dark Button
public class DarkButton implements Button {
@Override
public void render() {
System.out.println("Rendering a Dark Button");
}
}
// Concrete Product: Light TextField
public class LightTextField implements TextField {
@Override
public void render() {
System.out.println("Rendering a Light Text Field");
}
}
// Concrete Product: Dark TextField
public class DarkTextField implements TextField {
@Override
public void render() {
System.out.println("Rendering a Dark Text Field");
}
public interface UIFactory {
Button createButton();
TextField createTextField();
}
// Concrete Factory for Light Theme
public class LightUIFactory implements UIFactory {
@Override
public Button createButton() {
return new LightButton();
}
@Override
public TextField createTextField() {
return new LightTextField();
}
}
// Concrete Factory for Dark Theme
public class DarkUIFactory implements UIFactory {
@Override
public Button createButton() {
return new DarkButton();
}
@Override
public TextField createTextField() {
return new DarkTextField();
}
}
public class Application {
private Button button;
private TextField textField;
public Application(UIFactory factory) {
this.button = factory.createButton();
this.textField = factory.createTextField();
}
public void renderUI() {
button.render();
textField.render();
}
public static void main(String[] args) {
// Client can choose between different factories.
UIFactory factory = new DarkUIFactory(); // Could be switched to LightUIFactory
Application app = new Application(factory);
app.renderUI();
}
}
Output:
Spring Boot Example
In Spring Boot, the Abstract Factory pattern can complement dependency injection (DI) by delegating object creation logic to the factory. Here’s how to implement it with Spring Boot.
@Configuration
public class UIFactoryConfig {
@Bean
public UIFactory uiFactory(@Value("${app.theme}") String theme) {
if ("dark".equalsIgnoreCase(theme)) {
return new DarkUIFactory();
} else {
return new LightUIFactory();
}
}
}
@RestController
@RequestMapping("/ui")
public class UIController {
private final UIFactory uiFactory;
@Autowired
public UIController(UIFactory uiFactory) {
this.uiFactory = uiFactory;
}
@GetMapping("/render")
public void renderUI() {
Button button = uiFactory.createButton();
TextField textField = uiFactory.createTextField();
button.render();
textField.render();
}
}
In this example, the theme is configured through the application.properties
file, and the factory selection is handled by the Spring context.
Factory Method Comparison¶
Aspect | Factory Method | Abstract Factory |
---|---|---|
Purpose | Create one type of product. | Create families of related products. |
Complexity | Less complex. | More complex, involves multiple classes. |
Client Knowledge | Client knows about individual products. | Client works with factories, not specific products. |
Usage | Simple use-cases. | Complex, multi-product scenarios. |
Summary¶
The Abstract Factory Pattern is a powerful tool when designing systems that need to create multiple families of related objects. While it adds complexity, the benefits include extensibility, maintainability, and loose coupling. In a Spring Boot application, it works well alongside dependency injection, especially when configurations like themes or environments vary.