基本概念 工厂模式是一种创建型模式,它提供了一个创建对象的工厂,当开发者需要相关的对象实例时,由工厂来创建提供。简单来说,工厂模式可以根据不同的条件生产不同的实例,通常这些实例继承于同一个父类,工厂模式把创建这些实例的具体过程封装起来,简化了客户端的操作,也改善了应用的扩展性。
普通工厂模式 举一个使用工厂模式获取汪星人和喵星人的例子:
//创建二者共同的接口 public interface Animal { public void eat(); } //创建汪星人实现类 public class Dog implements Animal { @Override public void eat() { System.out.println("吃骨头"); } } //创建喵星人实现类 public class Cat implements Animal { @Override public void eat() { System.out.println("吃小鱼"); } } //创建工厂类 public class AnimalFactory { public Animal getAnimal(String type){ if("dog".equals(type)){ return new Dog(); }else if("cat".equals(type)){ return new Cat(); }else{ return null; } } } //测试 Animal dog = new AnimalFactory().getAnimal("dog"); 直接获取模式 由于普通工厂模式会因为传错字符串导致无法获得实例,故使用直接获取模式 //将创建工厂类修改 public class AnimalFactory { public Animal getDog(){ return new Dog(); } public Animal getCat(){ return Cat(); } } //测试 不用传入字符串,可直接获取 Animal dog = new AnimalFactory.getDog(); 静态工厂方法 将多个工厂方法设置为静态的,不用创建工厂实例,直接获取 public class AnimalFactory { public static Animal getDog(){ return new Dog(); } public static Animal getCat(){ return new Cat(); } } //测试 不用 new AnimalFatory() 直接获取 Animal dog = AnimalFactory.getDog();总体来说,工厂模式适合:出现了大量的实例需要创建,并且具有共同的接口时,可以通过工厂方法模式进行创建。在以上的三种模式中,第一种如果传入的字符串有误,不能正确创建对象,第三种相对于第二种,不需要实例化工厂类,所以,大多数情况下,我们会选用第三种——静态工厂方法模式。
当然对于上述几种工厂模式,有一个不好的地方就是,新增一个类,需要对生产工厂(AnimalFactory)进行修改,极大的限制了程序的可扩展性。 为了解决这一问题,引入了抽象工厂模式:所谓抽象工厂模式,就是为每一个实例提供一个工厂方法,当需要新增一个类时,不需要修改原来的工厂方法,只需要新建一个对应的工厂即可。上代码:
//定义接口 public interface Animal { public void eat(); } //实现接口 public class Cat implements Animal { @Override public void eat(){ System.out.println("吃小鱼"); } } //实现工厂类 public class CatFactory implements Provider { @Override public Animal produce(){ return new Cat(); } } public interface Provider { public void produce(); } //测试 Provider provider = new CatFactory(); Cat cat = provider.produce();这样以后想新增Dog类 只需要让Dog实现Animal接口,并提供DogFactory实现Provider即可,不需要修改之前的代码。
之前一直有个疑问,为什么在定义dog的时候用
Animal dog = new AnimalFactory().getAnimal("dog");而不用
Dog dog = new AnimalFactory().getAnimal("dog");其实这就是为什么使用接口或者抽象类的问题了。由于JAVA具有继承与多态的特点,当一个类继承抽象类或实现接口,其父类或者接口能够接受子类或者实现接口者的实例。这样能够提高代码的扩展性。 比如:
ErHaDog dog = new AnimalFactory().getAnimal("dog");当主人不想养二哈,而想养一只金毛,那么只需要创建一直JinMaoDog并实现Ainmal接口,这样只需要将工厂中创建二哈的代码替换成金毛即可,其他都不用修改,大大提高了程序的扩展性。反之,需要修改:
JinMaoDog dog = new AnimalFactory().getAnimal("dog");使程序的耦合度太高
