首页 > 编程语言 >算法的封装与切换——策略模式(三)

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

时间:2024-11-25 10:00:05浏览次数:7  
标签:封装 配置文件 double price Discount 算法 切换 折扣 public

作者简介:大家好,我是码炫码哥,前中兴通讯、美团架构师,现任某互联网公司CTO,兼职码炫课堂主讲源码系列专题

联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬

码哥名言:学习必须往深处挖,挖的越深,基础越扎实!

24.3 完整解决方案

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

202304012109186051.png

在图24-2中,MovieTicket充当环境类角色,Discount充当抽象策略角色,StudentDiscount、 ChildrenDiscount 和VIPDiscount充当具体策略角色。完整代码如下所示:

    //电影票类:环境类
    class MovieTicket {
    	private double price;
    	private Discount discount; //维持一个对抽象折扣类的引用
    
    	public void setPrice(double price) {
    		this.price = price;
    	}
    
        //注入一个折扣类对象
    	public void setDiscount(Discount discount) {
    		this.discount = discount;
    	}
    
    	public double getPrice() {
            //调用折扣类的折扣价计算方法
    		return discount.calculate(this.price);
    	}
    }
    
    //折扣类:抽象策略类
    interface Discount {
    	public double calculate(double price);
    }
    
    //学生票折扣类:具体策略类
    class StudentDiscount implements Discount {
    	public double calculate(double price) {
    		System.out.println("学生票:");
    		return price * 0.8;
    	}
    } 
    
    //儿童票折扣类:具体策略类
    class ChildrenDiscount implements Discount {
    	public double calculate(double price) {
    		System.out.println("儿童票:");
    		return price - 10;
    	}
    } 
    
    //VIP会员票折扣类:具体策略类
    class VIPDiscount implements Discount {
    	public double calculate(double price) {
    		System.out.println("VIP票:");
    		System.out.println("增加积分!");
    		return price * 0.5;
    	}
    }

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

    import javax.xml.parsers.*;
    import org.w3c.dom.*;
    import org.xml.sax.SAXException;
    import java.io.*;
    class XMLUtil {
    //该方法用于从XML配置文件中提取具体类类名,并返回一个实例对象
    	public static Object getBean() {
    		try {
    			//创建文档对象
    			DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
    			DocumentBuilder builder = dFactory.newDocumentBuilder();
    			Document doc;							
    			doc = builder.parse(new File("config.xml")); 
    		
    			//获取包含类名的文本节点
    			NodeList nl = doc.getElementsByTagName("className");
                Node classNode=nl.item(0).getFirstChild();
                String cName=classNode.getNodeValue();
                
                //通过类名生成实例对象并将其返回
                Class c=Class.forName(cName);
    	  	    Object obj=c.newInstance();
                return obj;
            }   
            catch(Exception e) {
               	e.printStackTrace();
               	return null;
           	}
        }
    }

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

    <?xml version="1.0"?>
    <config>
        <className>StudentDiscount</className>
    </config>

编写如下客户端测试代码:

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

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

原始价为:60.0

---------------------------------

学生票:

折后价为:48.0

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

    <?xml version="1.0"?>
    <config>
        <className>ChildrenDiscount</className>
    </config>

重新运行客户端程序,输出结果如下:

原始价为:60.0

---------------------------------

儿童票:

折后价为:50.0

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

标签:封装,配置文件,double,price,Discount,算法,切换,折扣,public
From: https://blog.csdn.net/smart_an/article/details/144017791

相关文章

  • 【完美复现】基于多智能体系统一致性算法的电力系统分布式经济调度策略(Matlab代码实现
    ......
  • 基于多目标哈里斯鹰算法及模型预测控制(MPC)的储能和风电平抑波动研究(Matlab代码实现)
     ......
  • 秦九韶算法
    这应该是笔者写的第一道多项式有关的题目没办法,我实在是太菜了就是这道题题面已经写的很露骨了但是如果你拿传统的枚举每个i去带入,并且你使用了快读+吸氧等一系列的优化,你就可以获得30pts的好成绩如果你开了高精度,那就是50pts所以正解就是秦九韶算法(第一眼我还以为是海伦—......
  • 【机器学习】决策树算法原理详解
    决策树1概述1.1定义决策树是一种解决分类问题的算法,决策树算法采用树形结构,使用层层推理来实现最终的分类。决策树即可以做分类,也可以做回归。它主要分为两种:分类树和回归树。1.2决策树算法第一个决策树算法:CLS(ConceptLearningSystem)使决策树受到关注、成为......
  • 蓝桥杯c++算法秒杀【6】之动态规划【上】(数字三角形、砝码称重(背包问题)、括号序列、
     下将以括号序列、组合数问题超级吧难的题为例子讲解动态规划别忘了请点个赞+收藏+关注支持一下博主喵!!!! ! ! ! !关注博主,更多蓝桥杯nice题目静待更新:)动态规划一、数字三角形【问题描述】        上图给出了一个数字三角形。从三角形的顶部到底部有很......
  • 蓝桥杯c++算法学习【5】之枚举与模拟(卡片、回文日期、赢球票、既约分数:::非常典型的比刷
     别忘了请点个赞+收藏+关注支持一下博主喵!!!! ! ! !!!关注博主,更多蓝桥杯nice题目静待更新:)枚举与模拟一、卡片:【问题描述】        小蓝有很多数字卡片,每张卡片上都是一个数字(0到9)。         小蓝准备用这些卡片来拼一些数,他想从1开始拼出正整数......
  • KuonjiCat的算法学习笔记:反悔贪心
    反悔贪心本蒟蒻在做题时被卡,看题解发现用反悔贪心,遂搜罗资料,得有此篇part.1什么是反悔贪心?简单的例子,我有一个只能装3个物品的背包,我要从n个价值由小到大的物品中选出3个最大的装进包里,但只能从头往后选,假如我此刻的包内物品价值为123,而我要面对的下一个物品的价值为4,那么......
  • 计算机视觉算法
    计算机视觉算法是一种使用计算机科学和数学方法来模拟人类视觉系统的算法。它涉及图像处理、模式识别和机器学习等技术,用于分析和理解图像或视频中的内容。常见的计算机视觉算法包括物体检测、图像分类、目标跟踪、人脸识别等。这里列举几种常见的计算机视觉算法:物体检测算......
  • 非机动车占道AI识别算法
    非机动车占道AI识别算法通过安装在交通道口的摄像头,非机动车占道AI识别算法运行AI识别算法对摄像头捕获的图像或视频流进行实时分析,识别非机动车占道行为。一旦检测到占道行为,系统会自动触发告警,并将信息发送至后台。接收告警信息,通知管理人员及时处理,并保存相关证据。该非机动车......
  • UnityShaderLab 实现两张图片切换
    实现思路:在顶点着色器中获取物体在世界空间下的某一轴的坐标传递给片元着色器,o.val=v.vertex.x;然后在片元着色器中采样两张图片,根据输入的偏移值的更换两张图片的显示。fixed4t1=tex2D(_MainTex,i.texcoord);fixed4t2=tex2D(_SubTex,i.texcoord1);......