第三次软件工程作业——两题
题目一
一、Github网址:
https://github.com/Rafael-Gu/The-3rd-Homework/tree/master/Q1
老师!!!老师!!!这是我问题一的作业!!!第二问我也做了,也是同一个网址。。。第一题是Q1的文件夹里面!!!
二、问题描述
题目(1):最大连续子数组和(最大子段和)
问题: 给定n个整数(可能为负数)组成的序列a[1],a[2],a[3],…,a[n],求该序列如a[i]+a[i+1]+…+a[j]的子段和的最大值。当所给的整数均为负数时定义子段和为0,依此定义,所求的最优值为: Max{0,a[i]+a[i+1]+…+a[j]},1<=i<=j<=n
例如,当(a[1],a[2],a[3],a[4],a[5],a[6])=(-2,11,-4,13,-5,-2)时,最大子段和为20。
-- 引用自《百度百科》
三、流程图
流程图如下:
四、程序源代码
程序源代码如下:
#include <stdio.h>
#include <algorithm>
using namespace std;
int sum(int *a,int j)
{
int last = 0, ans = 0, i = 0;
for (i = 1; i <= j; i++)
{
last = max(0, last) + a[i];
ans = max(ans, last);
}
return ans;
}
int main()
{
return 0;
}
五、条件组合覆盖测试
首先,程序中含有一个for
循环,其中for的条件中有一个条件需要满足i=j
此条件,其中for
调用了一次max
函数,而max
函数的具体逻辑如下:
由这两个流程图可知,此处有两个条件需要进行考虑,一个是for
中的i<=j
;另一个是max
函数中的判定条件。
六、单元测试代码:
namespace 作业三
{
TEST_CLASS(UnitTest1)
{
public:
TEST_METHOD(TestMethod1)
{
int a[] = { -1,-6,-3,-5,-1 };
Assert::AreEqual(sum(a, 5), 0);
}
TEST_METHOD(TestMethod2)
{
int *a = 0;
Assert::AreEqual(sum(a, 0), 0);
}
TEST_METHOD(TestMethod3)
{
int a[] = { -1,-6,-5,-3,-5,-2,6 };
Assert::AreEqual(sum(a, 7), 6);
}
TEST_METHOD(TestMethod4)
{
int a[] = { 0,-2,2,5,-9,0,7,5,13,-4,-6 };
Assert::AreEqual(sum(a, 11), 25);
}
TEST_METHOD(TestMethod5)
{
int a[] = { 1,-2,3,-4,5,-6,7,-8 };
Assert::AreEqual(sum(a, 8), 7);
}
};
}
此组数据,我采用了全为负数、正负数交替输入的几种类型,对程序进行单元测试,单元测试结果全部正确。
老师!老师!这是我另外做的问题一!问题二我也做了!问题一真的比问题二难啊!
这次作业学会了如何进行条件组合判定,以及强化了单元测试的操作
题目二
GitHub地址:
https://github.com/Rafael-Gu/The-3rd-Homework
一、题目要求
题目(2):下表为某商场每日营业额与应收税率的对照表,请编写一小程序根据该商场每日营业额计算其实际应缴纳税费。
营业额X (¥) | 1000≤X<5000 | 5000 ≤X<10000 | X≥10000 |
---|---|---|---|
税率 | 5% | 8% | 10% |
二、程序设计
1、测试代码如下:
namespace UnitTest1
{
TEST_CLASS(UnitTest1)
{
public:
TEST_METHOD(TestMethod1)
{
float x = 5000.00f;
float i = 200.00f;
Assert::AreEqual(jisuan(x), i, 0.01f);
x = 1000.00f;
i = 00.00f;
Assert::AreEqual(jisuan(x), i, 0.01f);
x = 10000.00f;
i = 600.00f;
Assert::AreEqual(jisuan(x), i, 0.01f);
x = 2000.00f;
i = 50.00f;
Assert::AreEqual(jisuan(x), i, 0.01f);
x = 6000.00f;
i = 280.00f;
Assert::AreEqual(jisuan(x), i, 0.01f);
x = 12000.00f;
i = 800.00f;
Assert::AreEqual(jisuan(x), i, 0.01f);
}
TEST_METHOD(TestMethod2)//侧重于测试异常情况
{
float x = NAN;
float i = 0.00f;
Assert::AreEqual(jisuan(x), i, 0.01f);
x = INFINITY;
i = 0.00f;
Assert::AreEqual(jisuan(x), i, 0.01f);
}
TEST_METHOD(TestMethod3)
{
float x = -1;
float i = 0.00f;
Assert::AreEqual(jisuan(x), i, 0.01f);
}
};
}
2、关于代码的思考
关于本题,我一开始只是将题目中的要求按照想法编了出来,我先对x先进行1000,5000,10000的输入,因为这三个数是关于交税的分界点,所以对这三个数进行测试,检查结果是否有误。然后找同学(大佬)来帮忙,作为浮点数x可能存在异常输入(NAN,INF),结果发现测试结果有误,发现程序中没有对这两个数进行考虑,导致单元测试没有通过。此外,对浮点数x进行负数输入进行单元测试,因为,负数在<1000的考虑范围之内,所以单元测试并未出现异常。程序代码如下:
#include <stdio.h>
#include <cmath>
float jisuan(float x)
{
if (isnan(x))
return 0;
if (isinf(x))
return 0;
if (x < 1000)
return 0;
else if (x < 5000)
return (x - 1000)*0.05;
else if (x < 10000)
return 200 + (x - 5000)*0.08;
else
return 600 + (x - 10000)*0.1;
}
int main()
{
float x;
scanf_s("%f", &x);
if (x < 1000)printf("0\n");
else if (x < 5000) printf("%.2f\n", (x - 1000)*0.05);
else if (x < 10000)printf("%.2f\n", 200 + (x - 5000)*0.08);
else printf("%.2f\n", 600 + (x - 10000)*0.1);
return 0;
}
3、流程图
三、覆盖标准测试
对于本次代码可知满足条件语句覆盖、判定覆盖、条件覆盖,其中,对于判定/条件覆盖,本代码中,每一个浮点数x都有一个对应的区间可带入到对应的函数中进行计算,得出最终的答案。条件组合覆盖满足条件,则是由边界点带入都有唯一一个函数带入得出答案。
1、判定/条件覆盖测试
x=NAN | x=INF | x<1000 | 1000≤x<5000 | 5000≤x<10000 | ≥10000 |
---|---|---|---|---|---|
return 0 | return 0 | return 0 | return (x - 1000)*0.05 | return 200 + (x - 5000)*0.08 | return 600 + (x - 10000)*0.1 |
2、样例测试
如上图可知,本次作业完成。
通过本次作业,我更加熟悉了单元测试的使用,同时也让我知道了如何更好地去进行程序测试,同时也提醒我要对输入值的范围和形式要考虑的更加周全。