设计模式之策略模式在地铁票价系统中的应用

2023-11-06 16:32

本文主要是介绍设计模式之策略模式在地铁票价系统中的应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

引言

设计模式是面向对象编程的一个非常精彩的部分。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性,它能帮助我们将应用组织成容易了解,容易维护,具有弹性的架构。本文通过一个简单的案例来讲述策略模式在地铁票价系统中的应用。

案例描述

乘客从一个车站乘坐地铁到另一个车站,他/她需要购买一张票。铁路部门对于票价有一些特别的票价规定:

按照市物价主管部门批复的轨道交通网络票价体系,即:轨道交通实行按里程计价的多级票价,0~6公里3元,6公里之后每10公里增加1元;票价计算采用最短路径法,即:当两个站点之间有超过1条换乘路径时,选取里程最短的一条路径作为两站间票价计算依据。

案例分析

让我们考虑有20个站点:1s,2s,3s......20s,并且乘客在不同的场景下乘坐地铁。为了更清晰的讲述问题,我们在原有定价标准上虚拟了一些应用场景。

  • 如果乘客A乘坐的里程小于6公里,那么他将需要支付3元车票费用。

  • 如果乘客B乘坐的里程大于6公里,他将需要额外支付超出部分的车票费用,计费标准为6公里之后每10公里增加1元。

  • 如果乘客C是VIP客户,那么他将在原计费标准上享受9折优惠。

  • 如果后续有一些额外收费或额外优惠,在以上计费基础上再进行调整。

解决方案

这个问题可以通过使用“策略设计模式”来解决。因为不同类型的票价策略可以基于不同的规则来应用。 以下是票价策略的不同类型:

  • 基本票价规则战略

  • VIP票价规则策略

  • 额外的票价规则策略

每张票价规则策略将分别写入票价计算算法,这些算法不会相互干扰。 新的票价规则可以添加和写入新的票价规则策略。这种模式也将遵循“对扩展开放、对修改关闭”的理念。

依赖关系图

类图

代码说明

IFareStrategy接口

这个接口定义了票价计算的常用策略,实现一个类可以实现基于上下文的票价算法。

using TrainFair.Models;    namespace TrainFair.FareCalculator  {  public interface IFareStrategy {  float GetFare(IFareRule ruleValues, float basicFare);  }  
}  

FareConstants类
FareConstants定义了计费的规则,包括起步价,超出里程递增价及VIP折扣价。

namespace TrainFair.Constants{    
public class FareConstants {        
public const float BasicFare = 3.0F;    
   public const float OnStationFare = 1.0F;  
        public const float VIPDiscount = 0.1F;} }

StationRuleFareCalculator类

StationRuleFareCalculator类根据行驶的车站里程和问题陈述部分定义的一些规则集来计算车费。

using System;  using TrainFair.Models;  namespace TrainFair.FareCalculator  {  public class StationRuleFareCalculator : IFareStrategy  {  public float GetFare(IFareRule ruleValues, float basicFare) {  var stationFareRuleModel = ruleValues as StationFareRuleModel;  if (stationFareRuleModel == null || stationFareRuleModel.StationDistance <= 0.0f)              
 return 0;            if (stationFareRuleModel.StationDistance < 6)                return basicFare;            int restChargingStations = (int)Math.Ceiling((stationFareRuleModel.StationDistance - 6.0f)/10.0f);            var totalFare = basicFare + restChargingStations * stationFareRuleModel.IncrementalPrice;           return totalFare;}  }   }  

VIPRuleFareCalculator类
这个类实现的是VIP的票价算法。如果乘客是VIP身份,那么他/她将得到享受特殊的优惠。这个类实现了这个算法。

using TrainFair.Models;  namespace TrainFair.FareCalculator  {  public class VIPRuleFareCalculator : IFareStrategy{       
      public float GetFare(IFareRule ruleValues, float basicFare) {            var vipFareRuleModel = ruleValues as VIPFareRuleModel;  
                if (vipFareRuleModel == null)        
                       return 0;           var totalFare = basicFare - (basicFare * vipFareRuleModel.Discount);            return totalFare;}} }  

OtherRuleFareCalculator类
这个类实现的是其他额外的费用或优惠票价的算法。一些额外的价格将被添加到总费用中。额外的价格可以是附加收费(正值),也可以是额外折扣(负值)。

using TrainFair.Models;  namespace TrainFair.FareCalculator  {  
 public class OtherRuleFareCalculator : IFareStrategy{      
   public float GetFare(IFareRule ruleValues, float basicFare) {                 var otherFareRuleModel = ruleValues as OtherFareRuleModel;              if (otherFareRuleModel == null)              
      return basicFare;          
        float totalFare = basicFare + otherFareRuleModel.AdditionalFare;            return totalFare;}} }  

FareRuleCalculatorContext类

using TrainFair.Models;  namespace TrainFair.FareCalculator  {  public class FareCalculatorContext {  private readonly IFareStrategy _fareStrategy;  public FareCalculatorContext(IFareStrategy fareStrategy) {  this._fareStrategy = fareStrategy;  }  public float GetFareDetails(IFareRule fareRules, float basicFare)  {  return _fareStrategy.GetFare(fareRules, basicFare);  }  }  
}  

代码结构里有一些基于车站票价,VIP票价,额外票价等情况的model类。

IFareRule接口

这是基本票价规则模型接口,每个模型类都实现它。

namespace TrainFair.Models  {  public interface IFareRule  {  int FareRuleId { get; set; }  }  
}  

StationFareRuleModel类
这个类定义的是车站票价规则的基本属性。

namespace TrainFair.Models  {  public class StationFareRuleModel : IFareRule  {  public int FareRuleId { get; set; }   
       public int  StationsCounts { get; set; }  
       public float IncrementalPrice { get; set; }        
       public float StationDistance { get; set; }  }   }

VIPFareRuleModel类

这个类定义了VIP折扣的属性。

namespace TrainFair.Models  {  
   public class VIPFareRuleModel : IFareRule  {  public int FareRuleId { get; set; }       public float Discount { get; set; }}   }  

OtherFareRuleModel类

这个类定义其他额外收费的属性。

namespace TrainFair.Models  {  public class OtherFareRuleModel : IFareRule  {  public int FareRuleId { get; set; }  public string OtherFareName { get; set; }  public float AdditionalFare { get; set; }  }  
}  

模型的属性可以根据未来的需求进行增强和调整,并可以灵活应用在算法类中。

执行结果

以下是控制台输出:

 
本文结尾附上了程序代码。

结语

车站基础票价、VIP票价、额外票价等不同类型的票价计算规则是不同的,所有的算法都被分解到不同的类中,以便能够在运行时选择不同的算法。策略模式的用意是针对一组算法或逻辑,将每一个算法或逻辑封装到具有共同接口的独立的类中,从而使得它们之间可以相互替换。策略模式使得算法或逻辑可以在不影响到客户端的情况下发生变化。说到策略模式就不得不提及OCP(Open Closed Principle) 开闭原则,即对扩展开放,对修改关闭。策略模式的出现很好地诠释了开闭原则,有效地减少了分支语句。

程序代码:https://github.com/daivven/TrainFair

原文地址:http://www.cnblogs.com/yayazi/p/8350679.html


.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

这篇关于设计模式之策略模式在地铁票价系统中的应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/357778

相关文章

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

基于人工智能的图像分类系统

目录 引言项目背景环境准备 硬件要求软件安装与配置系统设计 系统架构关键技术代码示例 数据预处理模型训练模型预测应用场景结论 1. 引言 图像分类是计算机视觉中的一个重要任务,目标是自动识别图像中的对象类别。通过卷积神经网络(CNN)等深度学习技术,我们可以构建高效的图像分类系统,广泛应用于自动驾驶、医疗影像诊断、监控分析等领域。本文将介绍如何构建一个基于人工智能的图像分类系统,包括环境

水位雨量在线监测系统概述及应用介绍

在当今社会,随着科技的飞速发展,各种智能监测系统已成为保障公共安全、促进资源管理和环境保护的重要工具。其中,水位雨量在线监测系统作为自然灾害预警、水资源管理及水利工程运行的关键技术,其重要性不言而喻。 一、水位雨量在线监测系统的基本原理 水位雨量在线监测系统主要由数据采集单元、数据传输网络、数据处理中心及用户终端四大部分构成,形成了一个完整的闭环系统。 数据采集单元:这是系统的“眼睛”,

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

zoj3820(树的直径的应用)

题意:在一颗树上找两个点,使得所有点到选择与其更近的一个点的距离的最大值最小。 思路:如果是选择一个点的话,那么点就是直径的中点。现在考虑两个点的情况,先求树的直径,再把直径最中间的边去掉,再求剩下的两个子树中直径的中点。 代码如下: #include <stdio.h>#include <string.h>#include <algorithm>#include <map>#

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)