软工2019作业三

吉哈地址:

>>>https://github.com/DreamFeather/031702113<<<

PSP表格:


PSP2.1 Personal Software Process Stages 预估耗时(分钟) 实际耗时(分钟)
Planning 计划 15 15
Estimate 估计这个任务需要多少时间 15 15
Development 开发 320 660
Analysis 需求分析 (包括学习新技术) 20 120
Design Spec 生成设计文档 0 0
Design Review 设计复审 30 0
Coding Standard 代码规范 (为目前的开发制定合适的规范) 0 0
Design 具体设计 20 120
Coding 具体编码 120 150
Code Review 代码复审 10 30
Test 测试(自我测试,修改代码,提交修改) 120 240
Reporting 报告 160 270
Test Report 测试报告 20 120
Size Measurement 计算工作量 20 120
Postmortem & Process Improvement Plan 事后总结, 并提出过程改进计划 120 30
合计 495 945



解题思路

看到题目是数独的时候,我大脑里第一反应是,游戏,数学家没事时玩的,一张的纸,一支铅笔,擦擦写写。好吧,这个游戏我听说过,但是从来没玩过,所以做的第一件事,在手机上装个数独游戏玩玩。这里我推荐“数独专业版”app,没有广告,界面简洁,小米商店评分4.9,下载来体验一把做数学家的惊险与刺激,简直是不二选择。边玩边思考,玩了两盘,灵感就来了。


首先,解一个数独题,其实就三步走:
第一,是最基本的,要知道哪些格子里有数,哪些格子没数。
第二,是最关键的,没数的格子里可以填哪些数。
第三,是最重要的,如何把格子填满。
第一步,用一个很简单的if语句就可以判断出哪些格子有没有数,没有数的要记录下来。我把它们放到一个队列里,排队等候填数

for (int i = 0; i != max; ++i)
    {
        number[i] = new int[max];
        for (int j = 0; j != max; ++j)
        {
            number[i][j] = array[i][j];                                                        //这里其实是Koe(宫格)类的初始化过程
            if (number[i][j] == 0)space_x.push(i), space_y.push(j);        //顺便找一下待处理的格子,将其坐标存入队列space
            else if (divided)block[int(i*div_x)][int(j*div_y)][number[i][j]] = 1;//划分块,没宫的用不着
        }
    }

第二步,如果一个格子没数,如何得到它能填的数呢?从游戏规则上来讲是横竖不重复,分块内不重复。那就得从已存在的数入手

void Koe::available(int i, int j, queue<int> &rest)                //找寻i行j列元素可用数,存放在rest队列
{
        int m = 0, max_ = max + 1;
    int *exist = new int[max_];                                            //因为要以存在数的值作为下标,所以得多开一点空间
    while (m != max_)exist[m++] = 0;
    for (m = 0; m != max; ++m)
    {
        exist[number[i][m]] = 1;                                        //横竖同时判断,一个循环搞定
        exist[number[m][j]] = 1;                                        //不用跳过自己,反正必定有number[i][j]=0,再加判断只是空耗开销
    }
    if (divided)                                                                    //从分块里再看
    {
        int x = int(i *div_x), y = int(j *div_y);                      //用i,j乘以分块划分比,即可得出i,j所在分块下标
        for (int z = 1; z != max_; ++z)
        {
            if (block[x][y][z] == 1)exist[z] = 1;                   //block[x][y][z]=1的意思是,分块[x][y]中存在数字z
        }
    }
    m = 1;                                                                            //从1开始记录
    while (m != max_)
    {
        if (exist[m] == 0)rest.push(m);                                //不存在的放入可用队列rest
        ++m;
    }
    delete []exist;                                                                //new出来的数组可以删了
        exist=NULL;
}

第三步,怎么把格子填满?我用的是递归的方法,逐个处理在Koe初始化的时候,我已经把空位存入了队列space,所以我只要一个一个取出来填就行了,填什么?上面的的available方法已经给了答案(以下源码经过简化)

int Koe::deduce(queue<int>s_x,queue<int>s_y)
{
        int x = s_x.front(), y = s_y.front();
    queue<int> rest;
    available(x, y, rest);                                                //就当前位置找可用数字
    s_x.pop(), s_y.pop();
    int blk_x = int(x *div_x), blk_y = int(y * div_y);
    int record = number[x][y];                                        //记录当前数字,保存现场。很没必要,我知道它一定是0
    int answer = 0;
    while (!rest.empty())                                                //可用数字不为空,就一直找下去
    {
        number[x][y] = rest.front();                                //填一个数字
        rest.pop();
        if (divided)block[blk_x][blk_y][number[x][y]] = 1;    //填好数字后相应的分块里要置1,表示占用
        if (!s_x.empty())answer += deduce(s_x, s_y);        //进入下一阶填空
        ......
        if (divided)block[blk_x][blk_y][number[x][y]] = 0;     //分块内数字取消占用
    }
    if (s_x.empty())                                                        //空填完了,即找到了答案
        {
            ......
        }
    number[x][y] = record;                                            //恢复数字,等于0即可
    return answer;
}



整个项目,只有Koe一个类,里面装的有点多,是项目的主体,结构如下




很多人会这觉得,解出数独即是完成了这次作业。解数独确实是这个项目的主要需求,也是整个项目构建起手之处,当然不排除有些人先构建IO啦。我是从解数独Koe类开始的,但是从新建一个项目到屏幕上可以正确输出只过了40分钟,准确地说43分钟,比我预想的要快得多。我想了想,我做完了吗?没有。停下来思考一下,总体完成度大概在66%左右。


这次作业还有一个关键点在于——对命令行的输入处理
并且,在作业要求里也有明确提到,对于错误的处理。我想了想,对于错误处理,在内部数据结构正确的情况下,出错基本是因为对输入处理得不够严谨,用正确的算法处理错误的数据,当然不会得到正确得结果。
我的目标是:对于任何输入,我的程序都能够有相应的反应。

  • 只要命令参数里叙述的信息逻辑正确,符合规定,我就一定能提取出正确有效的信息。比如规定-m后是数独阶数,那么-m 3,3在-m后面,我就知道了,3带代表求解的数独阶数。
  • 只要从命令行里能得到足够的、有效的输入信息,我的程序就一定能输出正确结果。
  • 得不到足够的、有效的信息,或者无法识别输入信息,一定要有相应的错误提示,告诉用户有错误,可能错在哪里。

错误处理考虑:

  • 命令行参数个数,最基本是要考虑到是有9个参数,Sudoku.exe -m 宫格阶数 -n 数独盘数 -i 输入文件 -o 输出文件,不是你期望的个数肯定就错了。我的为了实现多点的功能,参数有9,10,11三种可能。
  • 命令行参数顺序,作业要求的原话是:“从-m之后获取盘面阶数,-n之后获取盘面数量,-i之后获取输入文件名,-o之后获取输出文件名。”输入顺序我不知道,我只知道从某个命令后获取到的数据代表什么,只要信息逻辑表达正确,我就能获取到正确有效的信息。
  • 参数读到程序里来了,虽然正确有效符合逻辑,但还得确认它在我程序能处理的范围内。宫格阶数,整数范围[3,9];数独盘数,整数大于0就好;输入文件,首先能打开就好;输出文件的话,要求不高,不要和我的命令(-m,-n,-i,-o等)重名,(输出文件路径的问题有待考虑)。
  • 命令行的参数考虑到这里。另一个输入是在文件里,首先,我已经找到了,但是里面的数据不对,我肯定也处理不了。所以,先判断,文件有没有足够的内容,在读的过程中,要检查是否读到文档末尾了,我没读完就没了,那肯定不对,要报错,不是阶数错了就是文件错了。
  • 文件内容人眼看上去是够的,5X5矩阵,9X9矩阵,非常整齐。但是万一其中插入了非数字字符呢?我测试过,fstream对象输入字符数据到int类型,程序可以是直接挂的,当场闪退。用户一脸黑人问号:闪一下就没了???辣鸡软件!!!用户肯定不会想到是自己的输入文件有问题,尤其还是那种喜欢把测试用例文件改来改去的(比如我室友 (눈_눈!) ),打上了一个字符也毫不知情,后面测试两小时你其实能想到我们在干嘛了,简直害skr人。好吧,在文件里检测到字符了或者其他非数字的输入,报错,精准到几行几列,检查盘数的时候输出现在是检查第几盘,所以报错后我们能迅速找到错误。

对于输入的处理我还不敢保证绝对完美,毕竟我个人的精力和脑回路是有限的,我暂时已经想不出还会有其他我没考虑到情况了。实际上我可能已经过度考虑了:虽然atoi函数只能返回int型,但要是阶数盘数输入出现小数我也会输出警告。此时,虽然程序已经获得到了可用有效的信息,但是我还是遗憾地选择终止进程,因为输入不符合规范,规范是整数。比如-m 3.8,我用atoi转换一下只能得到3,但是在*argv[]里我仍然能找到一个'.',基于人性化考虑我可以让它以3接着运行下去;基于严谨性我认为我获得了一个有效信息3,可是这个有效信息3和用户输入的原始信息3.8所表达的信息有出入。我该如何在这两者之间选择或者权衡?我最终还是选择了严谨。
再到头来看,可能会有人觉得可笑了,因为用户输入小数的可能性很小啊,用户自己是想输入整数的,他也清楚自己要是输入小数那绝对也不对,总之输入的可能性超级小,而且还有atoi替我处理......好吧,不讨论这个了,越讨论越觉得这个处理没必要。


可改进的地方

在写博客的过程中,我能想到很多在之前没有想到的事。

  1. 我觉得写代码的同时加上注释是一个十分重要的规范。在这里我把它当作规范而不是说好习惯,因为习惯看个人,规范看集体。我之前就把写注释当作习惯来看待,好坏与否我不是特别在乎。但是现在,我必须得注意了,除了有时候我自己看糊涂外,也是为了别人能更快地理解并读懂。写博客,大大地增加了我们代码的曝光度,写好注释,很重要。
  2. 代码结构还有需要改进的地方,首先是数独找0处理,我觉得还有优化的地方。我现在的处理方式是,从(0,0)扫瞄到(m,m)处,那么找出来的0的坐标在space队列里也是沿从坐到右,从上到下排列下来的,之后我直接就用这个顺序来执行递归搜索了。我想到的更优的算法是,将这个space队列里的坐标以该处可用数字个数从少到多进行一次排序,也就是可用数字更少的先填,不然其他的地方占用了,递归就逐级返回搜索,直到将占用的地方改掉,才能继续往下走。毕竟题目要求只要一个解,最好的情况就是一路填下去填到最后一个space,把可用数字少的地方先填了,大概率实现所谓的一次找到,就算没有找到,递归返回的次数也会更少,这样代码性能能得到极大优化。
  3. 我还是有点纠结那个命令行输入的操作,小数就不讨论了,我觉得能优化的地方是对于输出文件怎么决定,因为对于目前来说我在-o后面随便打什么(除了命令),它都能接受,比如-w,甚至乱码,dgh1214,它打不开它也能新建一个,而且文件用txt格式打开后也一字不差,这样太随意了。我目前能想到的就是判断输出文件路径最后四个字符是不是.txt,不是就不行。或者我可以人性化一点给它加上后缀?但是软工老师在课上说过这么一句话:“不是用户提出来的要求不要去做,做了白费精力。”所以嘞,考虑到严谨性我还是稍微做一下规范,仅输出一句提醒,不终止进程,仅此而已。

代码分析

代码分析还是不会用,就算用出来了我也看不懂,以下,多图警告!(→ܫ→)

递归算法,算是十分形象了



猜你喜欢

转载自www.cnblogs.com/M031702113/p/11577097.html
今日推荐