第二次博客作业00

《构建之法》第二次博客 熟悉工具

一、作业要求

GIT地址 GIT地址
GIt用户名 Cherish599
学号后五位 62322
我的博客地址 博客

二、项目背景

阿超家里的孩子上小学一年级了,这个暑假老师给家长们布置了一个作业:家长每天要给孩子出一些合理的,但要有些难度的四则运算题目,并且家长要对孩子的作业打分记录。

作为程序员的阿超心想,既然每天都需要出题,那何不做一个可以自动生成小学四则运算题目与解决题目的命令行 “软件”呢。他把老师的话翻译一下,就形成了这个软件的需求:

程序接收一个命令行参数 n,然后随机产生 n 道加减乘除(分别使用符号+-*/来表示)练习题,每个数字在 0 和 100 之间,运算符在 2 个 到 3 个之间。

由于阿超的孩子才上一年级,并不知道分数。所以软件所出的练习题在运算过程中不得出现非整数,比如不能出现 3÷5+2=2.6 这样的算式。

练习题生成好后,将生成的 n 道练习题及其对应的正确答案输出到一个文件 subject.txt 中。

当程序接收的参数为4时,以下为一个输出文件示例。

13+17-1=29

11*15-5=160

3+10+4-16=1

15÷5+3-2=4

三、环境配置

Visual Studio 2013是我大一的时候就安装好的,版本不是很高,但是觉得再安装一个Visual Studio 2017或者Visual Studio 2019很浪费时间,同时也增加了电脑的负担。所以这次四则运算的作业我还是用的Visual Studio 2013来实现的。如果没有的可以去官网下载或者在网上找百度云盘的资源。

同时github也是以前就开始用了,当然注册一个账号也不是什么难事,有的博客说最好用谷歌浏览器来注册,我记得当时没有用谷歌,也没遇到什么问题。所以我觉得用什么浏览器应该没关系吧,当然最好用谷歌浏览器嘛。

对于git的安装,在Git的官网https://git-scm.com/点击Downloads

第二次博客作业00

下载git 2.23

第二次博客作业00

下载完成后,点击该应用程序,跟着引导一步一步的点击Next就OK了。git的安装十分简单,应该不需要什么安装教程,如果有需要的可以参考:https://www.cnblogs.com/wj-1314/p/7993819.html

四、项目克隆

登录自己的github账号,登陆成功后,输入阿超的仓库地址:

https://github.com/Cherish599/AchaoCalculator,进入阿超四则运算生成器仓库,点击右上角的Fork键,将该仓库拷贝到自己的同名仓库中。

第二次博客作业00

回到自己的首页中,查看是否拷贝成功。如果自己主页下面有“阿超四则运算生成器”,则说明拷贝成功,否则还需要重新拷贝。

第二次博客作业00

点击该页面右下角的Clone or download会出现一个网址,将该网站复制下来。

在想要下载该项目的地方,鼠标右键点击Git Bash Here,在给窗口下输入git clone并且将刚刚复制的网址粘贴上去。

第二次博客作业00

在保存的地方会出现刚下载的文件。在此我们可以新建自己的项目保存在这个文件夹中。

第二次博客作业00

五、创建项目

打开自己的Visual Studio,在开始栏下点击新建项目。

第二次博客作业00

选择Visual C#的控制台应用程序,将该项目保存在刚刚下载的四则运算文件夹下。如果不熟悉C#的话,也可以用C++语言来编程。

第二次博客作业00

代码块:
产生随机数的方法

public class Rand
{
    //产生随机数
    public static Random rand = new Random();
    /*
    public int n1 = rand.Next(0, 100);
    public int n2 = rand.Next(0, 100);
    public int n3 = rand.Next(0, 100);
    public int n4 = rand.Next(0, 100);
    public int s1 = rand.Next(0, 4);
    public int s2 = rand.Next(0, 4);
    public int s3 = rand.Next(0, 4);
    public int type = rand.Next(0, 2);
    */
}

输入输出的方法

public class Input
{
    public static int input()
    {
        Console.WriteLine("请输入需要产生的题目数:");
        //题目数量
        int n = Convert.ToInt32(Console.ReadLine());
        return n;
    }
    //文本输出
    public static StreamWriter sw1 = new StreamWriter(@"E:\gitKU\AchaoCalculator\Loserhqs\Primarymath\Primarymath\result.txt");
}

衔接的方法,即中间过程

将运算的情况进行分类

public class Linkup
{
    //判断算式结果是否为整数
    public static bool nonnegint(double sum)
    {
        if (sum == (int)sum)
            return true;
        else
            return false;
    }
    //两个数的运算
    public static double arith(int a, int b, int s)
    {
        if (s == 0)
            return a + b;
        else if (s == 1)
            return a - b;
        else if (s == 2)
            return a * b;
        else
        {
            if (b != 0)
            {
                int temp = a / b;
                if (temp * b == a)
                    return a / b;
                else
                    return -1;
            }
            else
                return -1;
        }
    }
}


public class Opterator
{
    //两个运算符
    public static int optwo(int n1, int n2, int n3, int s1, int s2, char[] oper)
    {
        int flag = 0;
        //if ((oper[s1] == '+' || oper[s1] == '-') && (oper[s2] == '*' || oper[s2] == '/'))
        if ((s1 / 2) == 0 && (s2 / 2) == 1)
        {
            double sum = Linkup.arith(n2, n3, s2);
            if (sum >= 0)
            {
                if (Linkup.nonnegint(sum))
                {
                    double temp = Linkup.arith(n1, (int)sum, s1);
                    if (temp >= 0)
                    {
                        if (Linkup.nonnegint(temp))
                        {
                            flag = 1;
                            Console.WriteLine(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "=" + (int)temp);
                            Input.sw1.Write(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "=" + (int)temp + "\r\n");
                        }
                    }
                }
            }
        }
        else
        {
            double sum = Linkup.arith(n1, n2, s1);
            if (sum >= 0)
            {
                if (Linkup.nonnegint(sum))
                {
                    double temp = Linkup.arith((int)sum, n3, s2);
                    if (temp >= 0)
                    {
                        if (Linkup.nonnegint(temp))
                        {
                            flag = 1;
                            Console.WriteLine(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "=" + (int)temp);
                            Input.sw1.Write(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "=" + (int)temp + "\r\n");
                        }
                    }
                }
            }
        }
        return flag;
    }
    //三个运算符
    public static int opthree(int n1, int n2, int n3, int n4, int s1, int s2, int s3, char[] oper)
    {
        int flag = 0;
        //if ((oper[s1] == '+' || oper[s1] == '-') && (oper[s2] == '*' || oper[s2] == '/') && (oper[s3] == '*' || oper[s3] == '/'))
        if ((s1 / 2) == 0 && (s2 / 2) == 1 && (s3 / 2) == 1)
        {
            double sum = Linkup.arith(n2, n3, s2);
            if (sum >= 0)
            {
                if (Linkup.nonnegint(sum))
                {
                    double temp = Linkup.arith((int)sum, n4, s3);
                    if (temp >= 0)
                    {
                        double temp1 = Linkup.arith(n1, (int)temp, s1);
                        if (temp1 >= 0)
                        {
                            if (Linkup.nonnegint(temp1))
                            {
                                flag = 1;
                                Console.WriteLine(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1);
                                Input.sw1.Write(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1 + "\r\n");
                            }
                        }
                    }
                }
            }
        }
        else if ((s1 / 2) == 0 && (s2 / 2) == 1 && (s3 / 2) == 0)
        {
            double sum = Linkup.arith(n2, n3, s2);
            if (sum >= 0)
            {
                if (Linkup.nonnegint(sum))
                {
                    double temp = Linkup.arith(n1, (int)sum, s1);
                    if (temp >= 0)
                    {
                        double temp1 = Linkup.arith((int)temp, n4, s3);
                        if (temp >= 0)
                        {
                            if (Linkup.nonnegint(temp1))
                            {
                                flag = 1;
                                Console.WriteLine(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1);
                                Input.sw1.Write(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1 + "\r\n");
                            }
                        }
                    }
                }
            }
        }
        else if ((s1 / 2) == 0 && (s2 / 2) == 0 && (s3 / 2) == 1)
        {
            double sum = Linkup.arith(n3, n4, s3);
            if (sum >= 0)
            {
                if (Linkup.nonnegint(sum))
                {
                    double temp = Linkup.arith(n1, n2, s1);
                    if (temp >= 0)
                    {
                        double temp1 = Linkup.arith((int)temp, (int)sum, s2);
                        if (temp1 >= 0)
                        {
                            if (Linkup.nonnegint(temp1))
                            {
                                flag = 1;
                                Console.WriteLine(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1);
                                Input.sw1.Write(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1 + "\r\n");
                            }
                        }
                    }
                }
            }
        }
        else if ((s1 / 2) == 1 && (s2 / 2) == 0 && (s3 / 2) == 1)
        {
            double sum = Linkup.arith(n1, n2, s1);
            if (sum >= 0)
            {
                if (Linkup.nonnegint(sum))
                {
                    double temp = Linkup.arith(n3, n4, s3);
                    if (temp >= 0)
                    {
                        double temp1 = Linkup.arith((int)sum, (int)temp, s2);
                        if (temp1 >= 0)
                        {
                            if (Linkup.nonnegint(temp1))
                            {
                                flag = 1;
                                Console.WriteLine(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1);
                                Input.sw1.Write(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1 + "\r\n");
                            }
                        }
                    }
                }
            }
        }
        else
        {
            double sum = Linkup.arith(n1, n2, s1);
            if (sum >= 0)
            {
                if (Linkup.nonnegint(sum))
                {
                    double temp = Linkup.arith((int)sum, n3, s2);
                    if (temp >= 0)
                    {
                        double temp1 = Linkup.arith((int)temp, n4, s3);
                        if (temp1 >= 0)
                        {
                            if (Linkup.nonnegint(temp1))
                            {
                                flag = 1;
                                Console.WriteLine(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1);
                                Input.sw1.Write(n1 + "" + oper[s1] + "" + n2 + "" + oper[s2] + "" + n3 + "" + oper[s3] + n4 + "=" + (int)temp1 + "\r\n");
                            }
                        }
                    }
                }
            }
        }
        return flag;
    }
}

生成一定数量的题目

public class Produce
{
    //存放运算符号的数组
    public static char[] oper = new char[4] { '+', '-', '*', '/' };
    public static void produce(int n)
    {
        int i=0;
        //有n道题
        while(i<n)
        {
            //生成两个或三个运算符的题目
            int type = Rand.rand.Next(0, 2);
            if (type == 0)
            {
                //随机生成三个数
                int num1 = Rand.rand.Next(0, 100);
                int num2 = Rand.rand.Next(0, 100);
                int num3 = Rand.rand.Next(0, 100);
                //随机生成两种运算符号
                int index1 = Rand.rand.Next(0, 4);
                int index2 = Rand.rand.Next(0, 4);
                int flag=Opterator.optwo(num1, num2, num3, index1, index2, Produce.oper);
                if(flag==1)
                {
                    i++;
                }
            }
            else
            {
                //随机生成四个数
                int num1 = Rand.rand.Next(0, 10);
                int num2 = Rand.rand.Next(0, 100);
                int num3 = Rand.rand.Next(0, 100);
                int num4 = Rand.rand.Next(0, 100);
                //随机生成三种运算符号
                int index1 = Rand.rand.Next(0, 4);
                int index2 = Rand.rand.Next(0, 4);
                int index3 = Rand.rand.Next(0, 4);
                int flag=Opterator.opthree(num1, num2, num3, num4, index1, index2, index3, Produce.oper);
                if(flag==1)
                {
                    i++;
                }
            }
        }
    }
}

运行结果:

控制台输出:

第二次博客作业00

记事本TXT文件:

第二次博客作业00

六、单元测试与回归测试

点击项目解决方案,选择该菜单页中的添加,然后新建项目。

第二次博客作业00

选择Visual C#下的测试,然后选择单元测试项目。在.cs文件下编写单元测试的程序。

第二次博客作业00

如果没有加该项目的引用的话,在写测试代码时,调用该项目中的方法会报错,我们需要在解决方案中添加该项目的引用。

第二次博客作业00

当单元测试程序编写完成后,对项目中各个类进行单元测试,测试所编写的代码是否有错误。

如果有,找到相应的位置进行修改;如果没有则通过测试。

第二次博客作业00
第二次博客作业00

回归测试是测试在修改了原有代码的基础上还能不能运行通过,会不会影响程序的整体实现,步骤其实和单元测试相同。

第二次博客作业00

单元测试与回归分析的代码:

public class UnitTest1
{
    [TestMethod]
    public void TestMethod()
    {
        //TestMethod1();
        //TestMethod2();
        //TestMethod3();
        //TestMethod4();
        TestMethod5();
    }
    public void TestMethod1()
    {
        //测试随机数
        int testnum = Rand.rand.Next(0, 100);
        int testopr = Rand.rand.Next(0, 4);
        int testtype = Rand.rand.Next(0, 2);
        Console.WriteLine(testnum);
        Console.WriteLine(Produce.oper[testopr]);
        if (testtype == 0)
        {
            Console.WriteLine("三个数的运算");
        }
        else
        {
            Console.WriteLine("四个数的运算");
        }
    }
    //测试两个数的运算
    public void TestMethod2()
    {
        //testA=8*9
        double ResultA = 72;
        double TestA = Linkup.arith(8, 9, 2);
        if (ResultA == TestA)
        {
            Console.WriteLine("testA的测试结果正确");
        }
        else
        {
            Console.WriteLine("testA的测试结果错误");
        }
        //testB=20/40
        double ResultB = 0.5;
        double TestB = Linkup.arith(20, 40, 3);
        if (ResultB == TestB)
        {
            Console.WriteLine("testB的测试结果正确");
        }
        else
        {
            Console.WriteLine("testB的测试结果错误");
        }
        //testC=56/8
        double ResultC = 7;
        double TestC = Linkup.arith(56, 8, 3);
        if (ResultB == TestB)
        {
            Console.WriteLine("testC的测试结果正确");
        }
        else
        {
            Console.WriteLine("testC的测试结果错误");
        }
    }
    //测试结果是否为整数
    public void TestMethod3()
    {
        double testnum1 = 5.63;
        double testnum2 = 45;
        if(Linkup.nonnegint(testnum1))
        {
            Console.WriteLine("testnum1测试正确");
        }
        else
        {
            Console.WriteLine("testnum1测试错误");
        }
        if (Linkup.nonnegint(testnum2))
        {
            Console.WriteLine("testnum2测试正确");
        }
        else
        {
            Console.WriteLine("testnum2测试错误");
        }
    }
    //测试运算函数
    public void TestMethod4()
    {
        //三个数的运算
        //testop1=78-5*9
        //int resultop1 = 33;
        Console.WriteLine("预测结果为33");
        Console.WriteLine("真实结果如下:");
        int flag1 = Opterator.optwo(78, 5, 9, 1, 2, Produce.oper);
        if(flag1==1)
        {
            Console.WriteLine("结果正确");
        }
        else
        {
            Console.WriteLine("结果错误");
        }
        //testop2=23+12/4
        //int resultop2 = 26;
        Console.WriteLine("预测结果为26");
        Console.WriteLine("真实结果如下:");
        int flag2 = Opterator.optwo(23, 12, 4, 0, 3, Produce.oper);
        if (flag2 == 1)
        {
            Console.WriteLine("结果正确");
        }
        else
        {
            Console.WriteLine("结果错误");
        }
        //四个数的运算
        //testop3=5*9-42/7
        //int resultop3 = 39;
        Console.WriteLine("预测结果为26");
        Console.WriteLine("真实结果如下:");
        int flag3 = Opterator.opthree(5, 9, 42, 7, 2,1,3 ,Produce.oper);
        if (flag3 == 1)
        {
            Console.WriteLine("结果正确");
        }
        else
        {
            Console.WriteLine("结果错误");
        }
        //testop4=5-9+77/11
        //int resultop4 = 11;
        Console.WriteLine("预测结果为11");
        Console.WriteLine("真实结果如下:");
        int flag4 = Opterator.opthree(5, 9, 77, 11, 1, 0, 3, Produce.oper);
        if (flag4 == 1)
        {
            Console.WriteLine("结果正确");
        }
        else
        {
            Console.WriteLine("结果错误");
        }
    }
    //测试能否生成相应数量的题目
    public void TestMethod5()
    {
        int n = 5;
        Produce.produce(n);
    }
}

七、断点调试

在主函数中设置断点,然后可以按F11步入,逐语句调试。

第二次博客作业00

下面是在F11逐语句调试过程中的情况。

第二次博客作业00

八、效能分析

通过分析栏下的性能与诊断,我们可以看到该项目的调用关系树:

第二次博客作业00

调用方/被调用方的情况,函数名称,非独占/独占样本数。

第二次博客作业00

还可以喊道调用函数,当前函数以及已调用函数的情况。

第二次博客作业00

同时也可以看到进程的情况。

第二次博客作业00

所有函数的的情况:非独占样本数,独占样本数以及它们的百分比都可以清楚的看到。

第二次博客作业00

九、上传提交

在AchaoCalculator 文件夹下,右键鼠标,git bash here,输入git add . 将项目文件添加到版本控制中。

第二次博客作业00

因为没有绑定github 用户的原因,我需要git config的指令来绑定我的用户,如果绑定的的话,可以直接输入git commit -m以及相关的备注,进行提交修改。

下图为绑定用户的过程:

第二次博客作业00

下图为提交的过程:

第二次博客作业00

输入git push,上传到github上面去。

第二次博客作业00

打开自己的github,查看是否上传成功。如果上传成功的话,会出现自己的文件名。

第二次博客作业00

点击走上角的New pull requst,将该项目的代码上传到阿超四则运算的仓库中去。打开刚开始获取阿超四则运算仓库中是否有自己的文件。

第二次博客作业00

十、总结

1.在安装git过程中,没有什么意外。整体来说,安装软件也是没有什么难的地方。只是第一次用git,不是很清楚它的一些指令,需要看相关的资料才能交文件下载与上传。

2.在用C#编程的过程中,也没有什么大的问题。可能是因为这个题的难度我还能接受的原因吧。但是有一段时间没有用C#编程了,有些用法与Python都能搞混,我也真是垃圾。

3.在用C#编程的过程中一定要注意静态变量的使用,因为C#是没有全局变量一说的。

4.第一次使用单元测试和回归测试,觉得很棒。以后的学习工作中,要经常使用这一功能。

上一篇:第一章 简单工厂模式


下一篇:cdq分治浅谈