算法的封装与切换——策略模式(三)

    为了实现打折算法的复用,并能够灵活地向系统中增加新的打折方式,Sunny软件公司开发人员使用策略模式对电影院打折方案进行重构,重构后基本结构如图24-2所示:

    为了提高系统的灵活性和可扩展性,我们将具体策略类的类名存储在配置文件中,并通过工具类XMLUtil来读取配置文件并反射生成对象,XMLUtil类的代码如下所示:

    1. import org.w3c.dom.*;
    2. import org.xml.sax.SAXException;
    3. import java.io.*;
    4. class XMLUtil {
    5. //该方法用于从XML配置文件中提取具体类类名,并返回一个实例对象
    6. public static Object getBean() {
    7. try {
    8. //创建文档对象
    9. DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
    10. DocumentBuilder builder = dFactory.newDocumentBuilder();
    11. Document doc;
    12. doc = builder.parse(new File("config.xml"));
    13. //获取包含类名的文本节点
    14. NodeList nl = doc.getElementsByTagName("className");
    15. //通过类名生成实例对象并将其返回
    16. Class c=Class.forName(cName);
    17. Object obj=c.newInstance();
    18. return obj;
    19. }
    20. catch(Exception e) {
    21. e.printStackTrace();
    22. return null;
    23. }
    24. }
    25. }

    在配置文件config.xml中存储了具体策略类的类名,代码如下所示:

    1. class Client {
    2. public static void main(String args[]) {
    3. MovieTicket mt = new MovieTicket();
    4. mt.setPrice(originalPrice);
    5. System.out.println("原始价为:" + originalPrice);
    6. System.out.println("---------------------------------");
    7. Discount discount;
    8. discount = (Discount)XMLUtil.getBean(); //读取配置文件并反射生成具体折扣对象
    9. mt.setDiscount(discount); //注入折扣对象
    10. currentPrice = mt.getPrice();
    11. System.out.println("折后价为:" + currentPrice);
    12. }
    13. }

    编译并运行程序,输出结果如下:

    如果需要更换具体策略类,无须修改源代码,只需修改配置文件,例如将学生票改为儿童票,只需将存储在配置文件中的具体策略类StudentDiscount改为ChildrenDiscount,如下代码所示:

    1. <?xml version="1.0"?>
    2. <config>

    如果需要增加新的打折方式,原有代码均无须修改,只要增加一个新的折扣类作为抽象折扣类的子类,实现在抽象折扣类中声明的打折方法,然后修改配置文件,将原有具体折扣类类名改为新增折扣类类名即可,完全符合“开闭原则”。