工厂方法模式是我们常用的模式之一,我们经常在以下情景使用:
1. 客户类不关心使用哪个具体类,只关心该接口所提供的功能;
2.创建过程比较复杂,例如需要初始化其他关联的资源类,读取配置配置文件等。
3.接口有很多具体实现或者抽象类有很多具体子类时,你可能需要为客户代码写一大串if-else逻辑来决定运行时使用哪个具体实现或者具体子类。
4.不希望给客户程序暴露过多此类的内部结构,隐藏这些细节可以降低耦合度。
5.优化性能,比如缓存大对象或者初始化比较耗时的对象。
GOF为工厂方法模式给出的定义如下:
Define an interface for creating an object,but let subclasses decide which class to instantitate. Factory Method lets a class defer instantiation to subclasses.
为创建对象定义一个接口,让子类决定实例化哪个类。工厂方法让一个类的实例化延迟到子类。
核心的工厂类不在负责所有的产品的创建,将具体创建工作推迟到了子类,更好的支持“开—闭”原则。
工厂方法模式的四种角色:
A.抽象工厂角色,工厂方法模式的核心,与应用程序无关。
B.具体工厂角色,是实现抽象工厂接口的具体java类。
C.抽象产品角色,是所有具体角色的一个抽象类或公有接口。
D.具体产品角色,工厂方法模式创建的任何对象都是这个角色的实例,具体产品角色由一个具体的java类实现。
创建数据库连接对象:
有时候实例化某些对象可能会非常消耗资源,比如创建数据库连接。我们需要考虑缓存这些实例对象,以方便下次能够继续重用它们,我们可以在工厂对象里创建一个缓存池缓存这些实例对象。在J2EE WEB应用中,我们经常使用java.lang.ThreadLocal缓存数据连接对象。
1.Product.java
public interface Product {
}
2.Factory.java
public interface Factory {//An interface mainly to create Product objects
Product createProduct();
}
3.
public class ConcreteProduct implements Product {
}
4.
public class ConcreteFactory implements Factory {
@Override
public Product createProduct() {
return new ConcreteProduct();
}
}
5.
public class ConnectionFactory {
private final ThreadLocal<Connection> connections = new ThreadLocal<Connection>();
//other variables and methods...
public Connection currentConnection() {
Connection conn = connections.get();
if (conn == null) {//if no connection binds to this thread, create a new one
//create a new connection.
conn = createConnection();
//bind this connction to current thread
connections.set(conn);
}
return conn;
}
public void setCurrentConnection(Connection conn){
connections.set(conn);
}
public void closeConnection(Connection conn){
connections.remove();
}
private Connection createConnection() {
//create a connection. Eg, conn = dataSource.getConnection();
return null;
}
}
6.
public class Client {
private Factory factory;
public Client(Factory factory) {
this.factory = factory;
}
public void doSomething(){
Product product= factory.createProduct();
//to do something...
}
public static void main(String[] args) {
Client client = new Client(new ConcreteFactory());
client.doSomething();
}
}