这篇文件介绍使用设计模式中的策略模式和委托来解决多个IfElse判断语句和Switch语句,这种替换方式在其他语言也一样可以做到,比如PHP、JavaScript、Python或者Java等。
这里以C#为例进行演示。
需要为一个程序编写计算方法,根据标签名称来决定不同的计算,大体意思如下:
- if(标签名=="值1"){
- return sum(标签数据,附加标签数据);
- }
- if(标签名=="值2"){
- return max(标签数据,附加标签数据);
- }
- if(标签名=="值3"){
- return min(标签数据,附加标签数据);
- }
- if(标签名=="值4"){
- return 标签数据*blc+附加标签数据;
- }
- if(标签名=="值5"){
- return 标签数据+附加标签数据*blc;
- }
- ......
- if(标签名=="值21"){
- return 计算方法21;
- }
或者使用Switch语句:
- switch(标签名){
- case "值1":
- return sum(标签数据,附加标签数据);
- break;
- case "值2":
- return max(标签数据,附加标签数据);
- break;
- case "值3":
- return min(标签数据,附加标签数据);
- break;
- ......
- case "值21":
- return 计算方法21;
- break;
- default:
- return 0;
- break;
- }
因为设备很多,有很多没有列出的标签,后面可能增加的标签最多的时候有20多种,那么这个地方使用这样的方式就很不合适了。
先使用策略模式进行重构,再使用委托进行更加简明的重构。
针对这种问题容易想到的是设计模式中的策略模式,策略模式比较简单,我们可以将这部分计算封装在一个DLL文件中,在程序中调用,这样增加算法可以不必须改动主程序代码。
1、使用策略模式进行改写
策略模式就是封装算法,让算法独立于客户的使用,一般是在确定算法时直接调用对应的算法,我们可以进行改进,生产一个算法列表供选择。
主要是策略模式加算法列表,代码如下:
算法类:
- public abstract class YQCYFormula
- {
- protected List<double> InterListA;
- protected List<double> AddonListA;
- protected double RatioA;
- public string FormulaNameA;
-
- public void SetData(List<double> InterList, List<double> AddonList, double Ratio)
- {
- this.InterListA = InterList;
- this.AddonListA = AddonList;
- this.RatioA = Ratio;
- }
-
- public abstract double GetResult();
- }
-
- public class GetMax:YQCYFormula
- {
- public GetMax()
- {
- base.FormulaNameA= "最大值";
- }
-
- public override double GetResult()
- {
- double DResult = (this.InterListA.Union(AddonListA).ToList<double>()).Max();
- DResult = Math.Round(DResult, 2);
- return DResult;
- }
- }
-
- public class GetMin : YQCYFormula
- {
- public GetMin()
- {
- base.FormulaNameA = "最小值";
- }
-
- public override double GetResult()
- {
- double DResult = (this.InterListA.Union(AddonListA).ToList<double>()).Min();
- DResult=Math.Round(DResult,2);
- return DResult;
- }
- }
-
- public class GetSum : YQCYFormula
- {
- public GetSum()
- {
- base.FormulaNameA = "总和";
- }
-
- public override double GetResult()
- {
- double DResult = this.InterListA.Sum() + this.AddonListA.Sum();
- DResult = Convert.ToDouble(DResult.ToString("#0.00"));
- return DResult;
- }
- }
-
- public class GetAverage : YQCYFormula
- {
- public GetAverage()
- {
- base.FormulaNameA = "平均值";
- }
-
- public override double GetResult()
- {
- double DResult = (this.InterListA.Sum() + this.AddonListA.Sum()) / (this.InterListA.Count + this.AddonListA.Count);
- DResult = Convert.ToDouble(DResult.ToString("#0.00"));
- return DResult;
- }
- }
实际应用:
- ArrayList DictFormula = new ArrayList();//算法列表
- YQCYFormula MyFormula;//算法实例
-
- //注册算法
- MyFormula = new GetMax();
- DictFormula.Add(MyFormula);
- MyFormula = new GetMin();
- DictFormula.Add(MyFormula);
- MyFormula = new GetSum();
- DictFormula.Add(MyFormula);
- MyFormula = new GetAverage();
- DictFormula.Add(MyFormula);
使用:
- double Ratio = 0.1;
- List<double> InterList = new List<double>();
- List<double> AddonList = new List<double>();
-
- Ratio = 0.1;
- InterList = new List<double> { 1.1, 2.2, 3, 3, 4.4, 5.5 };
- AddonList = new List<double> { 1.5, 2.5, 3.5, 4.5 };
- textBox1.Text += comboBox1.Text+":"+ GetResult(comboBox1.Text, InterList,AddonList,Ratio).ToString()+Environment.NewLine;
因为算法是在独立的,这样如果有了新的算法,只需要增加算法并注册,调用程序并不需要改变,结果如下。

2、使用委托来改写。
主要使用算法字典和委托进行,代码如下:
定义算法字典:
Dictionary<string, FormulaYQCY> DictCalculation = new Dictionary<string, FormulaYQCY>();
定义算法:
- public delegate double FormulaYQCY(List<double> InterList, List<double> AddonList, double Ratio);
- public FormulaYQCY MyFormula;
-
- private double FormulaMax(List<double> InterList,List<double> AddonList,double Ratio)
- {
- double DResult = (InterList.Union(AddonList).ToList<double>()).Max();
- DResult = Math.Round(DResult, 2);
- return DResult;
- }
-
- private double FormulaMin(List<double> InterList, List<double> AddonList,double Ratio)
- {
- double DResult = (InterList.Union(AddonList).ToList<double>()).Min();
- DResult = Math.Round(DResult, 2);
- return DResult;
- }
-
- private double FormulaSum(List<double> InterList, List<double> AddonList, double Ratio)
- {
- double DResult = InterList.Sum() + AddonList.Sum();
- DResult = Convert.ToDouble(DResult.ToString("#0.00"));
- return DResult;
- }
-
- private double FormulaAverage(List<double> InterList, List<double> AddonList, double Ratio)
- {
- double DResult = (InterList.Sum() + AddonList.Sum()) / (InterList.Count + AddonList.Count);
- DResult = Convert.ToDouble(DResult.ToString("#0.00"));
- return DResult;
- }
注册算法:
- MyFormula = FormulaMax;
- DictCalculation.Add("最大值", MyFormula);
- MyFormula = FormulaMin;
- DictCalculation.Add("最小值", MyFormula);
- MyFormula = FormulaSum;
- DictCalculation.Add("总和", MyFormula);
- MyFormula = FormulaAverage;
- DictCalculation.Add("平均值", MyFormula);
使用:
- double DoubleReslut = 0.0;
- double Ratio = 0.1;
- List<double> InterList = new List<double> { 1.1, 2.2, 3, 3, 4.4, 5.5 };
- List<double> AddonList = new List<double> { 1.5, 2.5, 3.5, 4.5 };
-
- foreach (var item in DictCalculation)
- {
- if (item.Key == comboBox1.Text)
- {
- MyFormula = item.Value as FormulaYQCY;
- DoubleReslut = MyFormula(InterList, AddonList, Ratio);
- textBox1.Text += comboBox1.Text+":" + DoubleReslut.ToString()+Environment.NewLine;
- }
- }
实际输出:

在实际编写代码过程中,有了字典或者列表,一般情况下我们已经可以不再使用多If...else语句或者Switch语句来进行分支判断输出,代码和程序的灵活性也明显增加了。