反射在工厂模式上的应用
1.前言
之前写过一篇设计模式之简单工厂(Factory method),在这篇文章的“7.可配置的简单工厂实例”中,客户端没有传入参数,这是因为在factory中已经定义了需要读取的配置文件。但是这样有个缺点就是灵活性不够,必须明确指定需要读取配置文件中的某一项,比如上面就定义了必须读取的是配置文件中的ImplClass=edu.sjtu.erplab.yanmo.simplefactory.Impl2这一个条目,假设配置文件中有多个条目,我们想要通过客户端传入一个简单的参数ImplClass来动态调用,那么该实例是不能完成。
2.正文
2.1利用反射机制在客户端传入具体的"包.类名"动态创建实例
首先定义一个水果接口Fruit,里面有一个eat的方法
Fruit.java
package edu.sjtu.erplab.reflect; public interface Fruit { public void eat(); }
然后定义两类水果Apple和Orange继承Fruit接口
Apple.java
package edu.sjtu.erplab.reflect; public class Apple implements Fruit { @Override public void eat() { System.out.println("吃苹果"); } }
Orange.java
package edu.sjtu.erplab.reflect; public class Orange implements Fruit { @Override public void eat() { System.out.println("吃橘子"); } }
Factory.java
package edu.sjtu.erplab.reflect; public class Factory { public static Fruit getInstance(String className) { Fruit fruit=null; try { fruit=(Fruit)Class.forName(className).newInstance(); } catch (Exception e) { e.printStackTrace(); } return fruit; } }
最后在客户端中通过传入具体的类名来动态创建水果实例
FacotryDemo1.java
package edu.sjtu.erplab.reflect; public class FacotryDemo1 { public static void main(String[] args) { //通过工厂类去的接口实例,传入完整的包.类名。 Fruit f=null; f=Factory.getInstance("edu.sjtu.erplab.reflect.Apple"); f.eat(); f=Factory.getInstance("edu.sjtu.erplab.reflect.Orange"); f.eat(); } }
第一次传入的是Apple的类名,第二次传入的是Orange的雷鸣,所以两次运行eat方法结果是
吃苹果
吃橘子
2.2结合配置文件动态创建实例
以上操作代码虽然可以通过反射取得接口的实例,但是在操作的时候需要传入完整的包.类名称,而且用户也无法知道一个接口有多少个可以使用的子类,所以此时可以通过属性文件的形式配置所要的子类信息。
在接口类所在包下创建一个Fruit.properties文件,文件内容为:
apple=edu.sjtu.erplab.reflect.Apple
orange=edu.sjtu.erplab.reflect.Orange
在属性文件中用简单的apple和orange来表示完整的包.类名称,这样在使用时直接通过属性名称即可,不再需要一长串包.类名称。
要使用属性文件,我们需要创建属性操作类,具体实现如下:
package edu.sjtu.erplab.reflect; import java.io.IOException; import java.io.InputStream; import java.util.Properties; public class Init { public static Properties getPro() { Properties pro=new Properties(); InputStream in=null; try { in=Init.class.getResourceAsStream("Fruit.properties");//从此类所在的包下取资源 pro.load(in); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } finally{ try{ in.close(); }catch(IOException e){ e.printStackTrace(); } } return pro; } }
这个类用于读取该类所在的包下的配置文件Fruit.properties,并返回具体的配置文件中的内容。
然后再客户端中我们只需要通过配置文件操作类获取配置文件,并传入属性名称即可,代码实例如下
package edu.sjtu.erplab.reflect; import java.util.Properties; public class FactoryDemo2 { public static void main(String args[]) { Properties pro=Init.getPro(); Fruit f=null; //使用反射创建对象实例 f=Factory.getInstance(pro.getProperty("apple")); f.eat(); f=Factory.getInstance(pro.getProperty("orange")); f.eat(); } }
可以看到这里我们只需要往工厂方法中传入pro.getProperty("apple")即可,而pro.getProperty("apple")表示读取配置文件中key为apple所对应的value,也就是
edu.sjtu.erplab.reflect.Apple。在这里客户只需要知道apple而不需要知道edu.sjtu.erplab.reflect.Apple,实现了很好的封装。