C#使用策略模式或者委托替代多IfElse判断和Switch语句

  这篇文件介绍使用设计模式中的策略模式和委托来解决多个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语句来进行分支判断输出,代码和程序的灵活性也明显增加了。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值