This is the 8th day of my participation in the August More Text Challenge
Today let’s continue learning about the Factory pattern in Java design pattern, which is being updated.
Let’s learn about design patterns, say they are fundamental and fundamental, say they are not, and they are not. It’s here and there. Learn it well is also to let oneself further bar.
Like a sentence: “eight hours for life, eight hours for development”. ‘
Location: Shunhe Village, Lanshan County, Yongzhou City, Hunan Province
Author: Laugh with heart *😁
Design Mode series:
- Java Design pattern – singleton pattern
- Java Design Pattern – Factory Pattern (1) Simple Factory pattern
- Java Design Pattern – Factory Pattern (2) Factory method pattern
- Java Design Pattern – Factory Pattern (3) Abstract Factory pattern
- Java Design pattern – Builder pattern
- Java Design pattern – Proxy pattern
- Java Design pattern – Adapter pattern
- Java Design pattern – Decorator pattern
- Java Design pattern – Bridge pattern
- Java Design pattern – Appearance pattern
- Java Design pattern – Composite pattern
- Java Design pattern – Share meta pattern
- Java Design Pattern – Template method pattern
- Java Design pattern – Policy pattern
- Java Design pattern – Chain of Responsibility pattern
- Java Design Pattern – The Mediator pattern
- Java Design Pattern – Observer Pattern (publish/subscribe pattern)
- Continuously updated…
One, foreword
Let’s not worry about what the factory model looks like
Let’s take a look at the following example, how to design, how to write, to better.
Step by step, the Java Factory pattern is introduced.
Example 1)
Requirements: Design a coffee shop ordering system.
Design a Coffee class and define its two subclasses (AmericanCoffee and latte); Design a coffee shop (CoffeeStore), coffee shop has the function of ordering coffee.
The code is relatively simple, and I have introduced it step by step. If you don’t like it, you can read it directly below.
Let’s start by designing and coding the way we used to.
2) Class diagram relationship
3) Code implementation
Let’s do it the same way we did before.
1. Write the abstract class Coffee first
public abstract class Coffee {
public abstract void addMilk(a);
public abstract void addSugar(a);
public abstract String getName(a);
}
Copy the code
2, write good American Coffee and latte inherit Coffee abstract class
public class AmericanCoffee extends Coffee {
@Override
public void addMilk(a) { System.out.println("Milk your coffee."); }
@Override
public void addSugar(a) { System.out.println("Sugar your coffee."); }
@Override
public String getName(a) { return "Americano"; }}Copy the code
public class LatteCoffee extends Coffee {
@Override
public void addMilk(a) { System.out.println("Milk your coffee."); }
@Override
public void addSugar(a) { System.out.println("Sugar your coffee."); }
@Override
public String getName(a) { return "Latte"; }}Copy the code
3. Coffee shops
public class CoffeeStore {
public Coffee createCoffee(String type){
Coffee coffee = null;
if("americano".equals(type)) {
coffee = new AmericanCoffee();
} else if("latte".equals(type)) {
coffee = new LatteCoffee();
}
coffee.addMilk();
coffee.addSugar();
returncoffee; }}Copy the code
4. Write a client to test ordering coffee
public class Client {
public static void main(String[] args) {
CoffeeStore coffeeStore = new CoffeeStore();
Coffee coffee = coffeeStore.createCoffee("americano");
System.out.println(coffee.getName());
/** * Output: * Add milk to coffee * Add sugar to coffee * Americano */}}Copy the code
In fact, there is no problem at first glance, but if I need to add several kinds of coffee, what do you think is appropriate?
Is it necessary to modify the CoffeeStore code? And if we want to open meituan selling point list? How to change it?
In Java, everything is an object.
If we create the object directly new, it will seriously coupling the object, if we want to change the object, all the new objects need to be modified again, which obviously violates the open and close principle of software design. (And the repetitive work is killing me)
If we use factories to produce objects, we can only deal with factories and completely decouple objects. If we want to replace objects, we can directly replace objects in factories to achieve the purpose of decoupling from objects. So the biggest advantage of the factory model is decoupling.
Then came the simple Factory pattern (not one of the 23 classic patterns, but more of a programming habit). 😁
2. Simple factory model
2.1 Overview:
The simple Factory pattern is the creation pattern, also known as the Static Factory Method pattern, but it is not one of the 23 GOF design patterns. The simple factory pattern is one in which a factory object decides which instance of a product class to create. Simple factory pattern is the most simple and practical pattern in the factory pattern family, which can be understood as a special implementation of different factory patterns.
A simple factory contains the following roles:
- Abstract product: Defines the specification of the product, describing the main features and functions of the product. (Coffee in the example)
- Concrete product: A subclass that implements or inherits an abstract product (Americano, latte, etc., in the example)
- Concrete factory: provides a method to create a product by which the caller retrieves the product. (A factory to create objects)
Usage scenarios
The factory class is responsible for creating fewer objects;
The customer only knows the parameters passed into the factory class and doesn’t care how the object is created (logically);
2.2 Class Diagram Relationship:
Simply speaking is in the original design on the addition of a layer (there is no what is to add a layer to solve, not on the addition of two layers (dog head life)) 😁
2.3 Code Modification:
Some modifications have been made on the original basis:
Add a SimpleCoffeeFactory class where objects are created. 😀
public class SimpleCoffeeFactory {
public Coffee createCoffee(String type) {
Coffee coffee = null;
if("americano".equals(type)) {
coffee = new AmericanCoffee();
} else if("latte".equals(type)) {
coffee = new LatteCoffee();
}
returncoffee; }}Copy the code
Modify the CoffeeStore class again
public class CoffeeStore {
public Coffee createCoffee(String type){
SimpleCoffeeFactory factory = new SimpleCoffeeFactory();
Coffee coffee = factory.createCoffee(type);
coffee.addMilk();
coffee.addSugar();
returncoffee; }}Copy the code
It looks like you’re just giving the SimpleCoffeeFactory the power to create the object, but the SimpleCoffeeFactory factory class has decoupled the CoffeeStore class from the Coffee object, CoffeeStore no longer needs to be in charge of how specific product objects are created. It only needs to be responsible for its own business
On the other hand, of course, new coupling occurs, the coupling of the CoffeeStore object to the SimpleCoffeeFactory factory object, and the coupling of the factory object to the goods object.
If we added new varieties of coffee later, we would have to modify the SimpleCoffeeFactory code to violate the open and close principle.
The factory class may have many clients, such as creating Meituan takeout and so on, so only need to modify the factory class code, save other modification operations.
2.4 advantages and disadvantages
Advantages:
1) Encapsulates the process of creating objects, which can be directly obtained by parameters. Separating the creation of objects from the business logic layer eliminates the need to modify the customer code later, and makes it less likely to modify the customer code and easier to extend if the factory class is modified directly to implement a new product rather than in the original code.
2) The factory class decides which concrete class object should be created based on the information given by the outside world. By using the factory class, the outside world is freed from the embarrassment of directly creating concrete product objects and is simply responsible for “consuming” them. Regardless of how the objects are created and organized. It makes clear their respective responsibilities and rights, which is conducive to the optimization of the whole software architecture.
Disadvantages: As the number of specific product classes increases in the system, there may be a need for the factory class to create different instances based on different conditions. This kind of judgment on conditions and judgment on specific product types interlaced together, it is difficult to avoid the spread of module functions, is very adverse to the maintenance and expansion of the system, and violates the “open and closed principle”.
2.5. Extension – Simple static factory
Some people in development define the creation of objects in the factory class as static. This is the static factory pattern, and it is not one of the 23 design patterns. The following code
public class SimpleCoffeeFactory {
public static Coffee createCoffee(String type) {
Coffee coffee = null;
if("americano".equals(type)) {
coffee = new AmericanoCoffee();
} else if("latte".equals(type)) {
coffee = new LatteCoffee();
}
returncoffe; }}Copy the code
2.6. Extension – Simple factory + configuration file decoupling
Factory objects and product objects can be decoupled through the factory pattern + configuration file.
Load the full class name in the configuration file in the factory class and create objects for storage. If the client needs objects, it can directly obtain them.
-
Define a configuration file my.properties
american=com.crush.factory.simple_factory_properties.AmericanCoffee latte=com.crush.factory.simple_factory_properties.LatteCoffee Copy the code
-
Improved factory class
public class CoffeeFactory { private static Map<String,Coffee> map = new HashMap(); static { Properties p = new Properties(); InputStream is = CoffeeFactory.class.getClassLoader().getResourceAsStream("my.properties"); try { p.load(is); // Iterate over the Properties collection object Set<Object> keys = p.keySet(); for (Object key : keys) { // Get value by key (full class name) String className = p.getProperty((String) key); // Get the bytecode objectClass clazz = Class.forName(className); Coffee obj = (Coffee) clazz.newInstance(); map.put((String)key,obj); }}catch(Exception e) { e.printStackTrace(); }}public static Coffee createCoffee(String name) { returnmap.get(name); }}Copy the code
Static member variables are used to store created objects (keys store names and values store corresponding objects), while reading configuration files and creating objects are written in static blocks of code that are intended to be executed only once.
This method is also used a lot, common and very simple.
Three, endnotes
This simple factory is not perfect. When adding new products, we still need to modify the code of the factory class, which violates the “open and close principle”, so we have the factory mode and abstract factory mode in the following article.
It is constantly updated.