1.搭建ui,在Canvas中新建一个panel,新建两个InputField,作为两个输入框,获取输入的数值。新建四个button组件作为点击的按钮。新建一个Text组件作为计算结果的展示。界面搭建完成如下图

 2.编写代码

using System;
using UnityEngine;
using UnityEngine.UI;

namespace JiSuanQi.UI
{
    public class JISuanQiPanel : MonoBehaviour
    {
        public Button mBtnJiaFa, mBtnJianFa, mBtnChengFa, mBtnChuFa;
        public Text mResult;
        public InputField mInputNumA;
        public InputField mInputNumB;
        private void Awake()
        {
            mBtnJiaFa.onClick.AddListener(JiaFa);
            mBtnJianFa.onClick.AddListener(JianFa);
            mBtnChengFa.onClick.AddListener(ChengFa);
            mBtnChuFa.onClick.AddListener(ChuFa);
        }

        private void JiaFa()
        {
            double result = Operation(0);
            UpdateResult(result);
        }

        private void JianFa()
        {
            double result = Operation(1);
            UpdateResult(result);
        }

        private void ChengFa()
        {
            double result = Operation(2);
            UpdateResult(result);
        }

        private void ChuFa()
        {
            double result = Operation(3);
            UpdateResult(result);
        }

        private double Operation(int index)
        {
            if (string.IsNullOrEmpty(mInputNumA.text)||string.IsNullOrEmpty(mInputNumB.text))
            {
                return -1;
            }
            double NumA = double.Parse(mInputNumA.text);
            double NumB = double.Parse(mInputNumB.text);
            switch (index)
            {
                case 0:
                    return NumA + NumB;
                case 1:
                    return NumA - NumB;
                case 2:

                    return NumA * NumB;
                case 3:
                    if (NumB == 0)
                    {
                        return -1;
                    }
                    else
                    {
                        return NumA / NumB;
                    }
                default:
                    return -1;
            }
        }

        private void UpdateResult(double result)
        {
            mResult.text = "结果:" + result;
        }
    }
}

ui组件使用拖拽的方式进行赋值。ui组件赋值的方式一种是使用代码查找transform.Find("组件的ui面板路径").getComponent<组件类型>();在Awake或者Start方法中初始化。我比较喜欢拖拽的方式赋值。因为界面是经常改动的东西,使用的拖拽的方式赋值,如果ui删除了,代码部分并不需要修改。如果使用代码查找的方式,组件删除了,代码里查找的部分就会报错,会牵扯到代码部分的修改。不过用代码查找的方式可以在脚本中看到完整的方法绑定逻辑。容易阅读。

以上简单的加减乘除代码就完成了,开发中这种方式是最容易想到的,一个界面里就实现了所有的功能。

3.以上代码功能正常,但是是有问题的。界面业务逻辑都掺杂在一起了。但是界面会有很多种,如果现在新增了一个界面,同样要使用加减乘除的业务逻辑,但是界面不是计算器的界面。这时候就要重新写一边计算的逻辑,如果以后计算的逻辑改变了,那么两个界面都要修改计算的逻辑,这样既会增加工作量,还不能保证每次修改都是一样的,导致bug的产生。如果说现在要使用gui来实现同样的计算器功能,同样要把界面业务逻辑,复制一份到gui的代码中。每次更换ui界面的时候都会遇到同样的问题。

4.代码优化:ui逻辑业务逻辑实现分离

ui界面逻辑:

using System;
using UnityEngine;
using UnityEngine.UI;

namespace JiSuanQi.UI
{
    public class JISuanQiPanel : MonoBehaviour
    {
        public Button mBtnJiaFa, mBtnJianFa, mBtnChengFa, mBtnChuFa;
        public Text mResult;
        public InputField mInputNumA;
        public InputField mInputNumB;
        private TestOperation mOperation;
        private void Awake()
        {
            mOperation = new TestOperation();
            mBtnJiaFa.onClick.AddListener(JiaFa);
            mBtnJianFa.onClick.AddListener(JianFa);
            mBtnChengFa.onClick.AddListener(ChengFa);
            mBtnChuFa.onClick.AddListener(ChuFa);
        }

        private void JiaFa()
        {
            double result = Operation(0);
            UpdateResult(result);
        }

        private void JianFa()
        {
            double result = Operation(1);
            UpdateResult(result);
        }

        private void ChengFa()
        {
            double result = Operation(2);
            UpdateResult(result);
        }

        private void ChuFa()
        {
            double result = Operation(3);
            UpdateResult(result);
        }

        private double Operation(int index)
        {
            if (string.IsNullOrEmpty(mInputNumA.text)||string.IsNullOrEmpty(mInputNumB.text))
            {
                return -1;
            }
            double NumA = double.Parse(mInputNumA.text);
            double NumB = double.Parse(mInputNumB.text);
            return mOperation.GetResult(NumA, NumB, index);
        }

        private void UpdateResult(double result)
        {
            mResult.text = "结果:" + result;
        }
    }
}

业务逻辑:

namespace JiSuanQi
{
    public class TestOperation 
    {
        public double GetResult(double NumA,double NumB,int index)
        {
            switch (index)
            {
                case 0:
                    return NumA + NumB;
                case 1:
                    return NumA - NumB;
                case 2:

                    return NumA * NumB;
                case 3:
                    if (NumB == 0)
                    {
                        return -1;
                    }
                    else
                    {
                        return NumA / NumB;
                    }
                default:
                    return -1;
            }
        }
    }
}

把业务逻辑提取到了新的类TestOperation中,这样业务逻辑就抽取出来了。

以上代码,计算功能正常。

4.代码存在的问题:TestOperation类中维护了所有的计算逻辑。如果说后面新增了其他计算逻辑,如开根,求余数,等新的计算逻辑,或者对加减乘除的逻辑进行特殊的修改,都会牵扯到TestOperation类的修改,这样如果Operation类中出现了错误,会导致计算功能异常。并且不能保证每次修改都不会影响已使用的功能。因为修改的是TestOperation这个类。

5.代码优化:

using UnityEngine;

namespace JiSuanQi
{
    public abstract class Operation
    {
        public double NumberA;
        public double NumberB;
        public abstract double GetResult();
    }
}



namespace JiSuanQi
{
    public class OperationAdd:Operation
    {
        public override double GetResult()
        {
            return NumberA + NumberB;
        }
    }
}


namespace JiSuanQi
{
    public class OperationReduce:Operation
    {
        public override double GetResult()
        {
            return NumberA - NumberB;
        }
    }
}

namespace JiSuanQi
{
    public class OperationChengFa:Operation
    {
        public override double GetResult()
        {
            return NumberA * NumberB;
        }
    }
}

namespace JiSuanQi
{
    public class OperationChuFa:Operation
    {
        public override double GetResult()
        {
            return NumberA / NumberB;
        }
    }
}
using System.Collections;
using System.Collections.Generic;
using JiSuanQi;
using UnityEngine;

public class SimpleFactory 
{
   public static Operation CreateOperate(int index)
   {
      switch (index)
      {
         case 0:
            return new OperationAdd();
         case 1:
            return new OperationReduce();
         case 2:

            return new OperationChengFa();
         case 3:
            return new OperationChuFa();
         default:
            return null;
      }
   }
}

业务逻辑优化,将Operation类提取出来,加减乘除的操作作为一个对象,Operation类作为操作的基类,基类只要持有NumA和NumB两个字段和一个GetResult方法即可。具体的实现由字类重写。实现不同的逻辑。

SimpleFactory类。属于对象创建的工厂。根据输入的index不同,实例化不同的Operation。如果后期有修改,只需要修改对应的字类即可。

 

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐