http://www.jb51.cc/article/p-duxsenda-bao.html
同时,这篇博客其实也是对上文的再次扩展。
我们知道简单工厂模式的一个巨大的不足就是
系统扩展困难,一旦添加新产品就不得不修改工厂逻辑,在产品类型较多时,有可能造成工厂逻辑过于复杂,不利于系统的扩展和维护。
例如经典的工厂代码一般如下:
public class ChartFactory { //静态工厂方法 public static Chart getChart(String type) { Chart chart = null; if (type.equalsIgnoreCase("histogram")) { chart = new HistogramChart(); System.out.println("初始化设置柱状图!"); } else if (type.equalsIgnoreCase("pie")) { chart = new PieChart(); System.out.println("初始化设置饼状图!"); } else if (type.equalsIgnoreCase("line")) { chart = new LineChart(); System.out.println("初始化设置折线图!"); } return chart; } }一旦我增加一个新的chart,就得修改getChart这个类里面的逻辑。而且一堆if else看着实在难受。所以,我可以让每个chart自己判断。
看代码
package simplefactory; public interface Chart { //总的chart接口 public boolean match(String chart); public void display(); public void init(); }
package simplefactory; public class HistogramChart implements Chart { //柱状图的类,饼状图与折线图类似 不再赘述 public HistogramChart() { System.out.println("创建柱状图!"); } public void display() { System.out.println("显示柱状图!"); } public boolean match(String chart) { if (chart.equals("hisrogram")) return true; return false; } @Override public void init() { System.out.println("初始化设置柱状图!"); } }
在这里也有一个xml,如下
其中chartType是所有的chart类型
need 是我们所需要的chart类型
<?xml version="1.0" encoding="UTF-8"?> <config> <chartType>simplefactory.HistogramChart</chartType> <chartType>simplefactory.LineChart</chartType> <chartType>simplefactory.PieChart</chartType> <need>line</need> </config>
分析xml的工具类如下:
package xml; import javax.xml.parsers.*; import org.w3c.dom.*; import java.io.*; public class XMLUtil { // 该方法用于从XML配置文件中提取图表类型,并返回类型名 // 这个类和上文提到的博客里面的差别在于 方法的参数 public static String[] getChartType(String choice) { try { // 创建文档对象 DocumentBuilderFactory dFactory = DocumentBuilderFactory .newInstance(); DocumentBuilder builder = dFactory.newDocumentBuilder(); Document doc; doc = builder.parse(new File("src/xml/xmlutil.xml")); // 获取包含图表类型的文本节点 NodeList nl = doc.getElementsByTagName(choice); String[] names = new String[nl.getLength()]; for (int i = 0; i < nl.getLength(); i++) { names[i] = nl.item(i).getFirstChild().getNodeValue().trim(); } return names; } catch (Exception e) { e.printStackTrace(); return null; } } }
最后,我们看看重构后的工厂
package simplefactory; import java.util.ArrayList; import org.apache.catalina.tribes.membership.StaticMember; import xml.XMLUtil; public class ChartFactory { //产品候选列表 static ArrayList<Chart> charts=new ArrayList<Chart>(); //需要的产品 static String[] choise=new String[1]; public static Chart getChart(){ charts=ChartFactory.getAllChart(); choise=XMLUtil.getChartType("need"); for (int i = 0; i < charts.size(); i++) if (charts.get(i).match(choise[0])) return charts.get(i); return null; } public static ArrayList<Chart> getAllChart() { //获得所有chart的路径 String[] name = XMLUtil.getChartType("chartType"); ArrayList<Chart> chart = new ArrayList<Chart>(); try { //将chart全部new出来 for (int i = 0; i < name.length; i++) chart.add((Chart) Class.forName(name[i]).newInstance()); } catch (InstantiationException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IllegalAccessException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (ClassNotFoundException e) { // TODO Auto-generated catch block e.printStackTrace(); } return chart; } }
测试如下
package simplefactory; public class ChartTest { public static void main(String[] args) { Chart chart=ChartFactory.getChart(); System.out.println("------"); chart.init(); chart.display(); } }结果如下:
创建柱状图!
创建折线图!
创建饼状图!
------
初始化设置折线图!
显示折线图!
如果想要饼状图,只需要把xml改成
<need>pie</need>
我现在已经不能说这次重构的结果仍然是工厂模式了。工厂模式是生产一个匹配于所给参数的产品。而我这个等于是把工厂支持的产品先都生产出来,再一个一个问:人家需要的是这个产品,你看你行不?
好处就在于,以后即使要扩展新的产品,只用改xml就好。不用对工厂做改变。
而且我们告别了一大堆if else。
感谢glt