第一次进行真正的结对编程,而且我们组又是最奇葩的三人组合(14061183韩青长)(14061195陈彦吉),在经历了三天的合作以后,感觉收获还是蛮多的,下面是我对于结对编程的一些个人体验。
在查找这些设计方法的同时,我的内心是有些崩溃的,看着我写的程序,真的就是这些设计方法的反面典型啊。尤其是强耦合这一点,真的让我的结对伙伴超级头大。我们为了解耦花了不少的时间。不过相信通过这次的训练,今后对于程序设计方法上将会更加的熟悉。
当开发一个完整的程序时,可将程序的每个组成部分封装在一个模块中,每个模块尽量少地对外展示模块内部的数据与对数据的处理,以此提高代码的复用性、可维护性。减少外界可见的信息,以保持模块的独立性,以此降低耦合
从具体实现来说,我们确保某些成员变量和功能为private,并尽可能保证高层类只会调用底层类中特定的几个可供外界访问函数,而将其他方法封装在内部,限制了高层类对底层类的调用。
接口设计一共有六大原则。。原则说的都很清楚。。但是具体我们如何使用还是要多摸索。。
以下整理自博客
根据实际需求划分职责,尽量做到职责单一
所有引用基类的地方必须能透明地使用其子类的对象
高层模块不应依赖底层模块,两者都应依赖其抽象。抽象不应该依赖细节,细节应该依赖抽象
接口尽量小,不要建立大接口,同时接口中方法尽量少,将接口隔离起来,将整体框架进行有效划分。
跟information hiding有异曲同工之妙,此原则的使用可以降低程序耦合
一个软件实体如类、模块和函数应该对扩展开放,对修改关闭,编程前对不同的模块确立明确的逻辑分工,为变化预留位置
但是我们这一次由于大结构简单,因此没有设计接口,所以这些原则没有得到很好的实践。。希望能在以后的团队项目中积累相关经验
尽可能全面地把握耦合度的高低,平衡好耦合和内聚程度,以确保可读性、服用性、可维护性的提升
不可一味强调低耦合,例如程序中发生变更概率很小的地方,不宜强行低耦合。
还有程序中一些为大函数服务的小函数,小函数本身是依附大函数存在的,本身不会和其他模块建立过多的依赖关系,此时在耦合度上最好也做相应折衷
design by contract把类与被调用类之间的关系看作契约,规定了双方的权力与义务,这一方法被认为是构建OO软件系统方法的核心。
另外,
我们通过一些知乎回答了解到的契约式设计:
目的:
在设计程序时明确规定一个模块单元在调用前后应当处于何种状态,属于一种设计风格或是语法规范
思路:强调前置条件、后置条件和不变式,当违反这些操作时程序会抛出异常。
优点是:契约式编程使代码标准化、规范化,提高了程序工程化程度。
同时,对于测试者,在我们这次作业中,我们通过在不满足前置条件的情况下抛出异常,便于测试者测试。
缺点是有点麻烦。。
这可能是我之前比较自信的一部分了,不过没想到在结对最开始的时候我们其实一直在改我自己程序里之前的Bug,虽然不多但是真的非常难改啊!之前复审我代码的陈彦吉同学也说到了,我之前的代码因为采用了“用代码量换效率”的编码方式,三种路径计算用了三种不同的算法,背后需要的基本数据也不能公用。
也如我个人项目博客中说到的,我从最开始编码的构思阶段,到中途的实现阶段,再到最后的代码优化阶段,一直不停的在推翻我之前的算法,应用更新的,更高效的算法。
这一方面来讲是一件好事,因为确实效率不停的在提高:采用换乘站来抽象线路,搜索空间直接下降到了原来的20%,在巨量迭代下,整体效率提升巨大。利用迪杰斯特拉算法替代广搜在换乘站的图中进行寻路,又将时间开销下降了一个数量级。陈彦吉同学说我的算法比他快了50倍,我自己都没想到,本来也就觉得能快10倍左右的。
在另一方面来说,正如陈彦吉同学指出的,我的代码真的非常的长!而且是强耦合的模块化。这点我承认,因为光核心计算算法就有三个,奇葩的数据存储结构又需要用大量的辅助函数来对数据计算前的前期处理以及计算后的结果输出。其实这本来不是耦合性那么强的借口,但是我写的那些辅助函数为了节省代码量全部都进行了公用,在三种算法中都对几个辅助函数进行了反复调用,而且有些不便传值的数据:如地图等都直接写成了全局变量。放眼望去,真的是非常的冗长而且难以理解。
确实是因为之前没有开发这么长的应用程序的经验(最后看来整个程序接近3000行),感觉每时每刻都在不停地试错,最后完成的时候再回望最开始对于整个架构的设计,真是naïve。从现在来看,下次对于程序架构的设计应该更多的考虑模块之前的耦合程度,尽量做到松耦合和全面的模块化,尽量从一开始就选定好一个通用的数据结构,更多的考虑程序的扩展和维护的方便,写更多的注释让其他阅读程序的人能够读懂……凡此种种,这么多的经验教训,想来也是我这次个人项目和结对项目中的收获与进步吧。
关于算法的创新与效率上的提高,这两点我的个人项目总结也都写了,在开始结对项目之前,我也刚刚完成了个人项目的附加题(虽然要算将近一分钟),在这次作业中,我想我们组可能是为数不多的几个完成了三种搜索算法的组吧;)
这是我最骄傲的一部分了。个人给UI打90分!
先上效果图
设置面板
路径显示面板与动画
之前从来没有做过UI,不过也算看过很多UI了,自诩还有一些审美水平的我当然不想做一个会被人吐槽“丑”的UI。在了解了一下C#的各种UI制作方法后,我抛弃了陈年老旧的WinForm,选择了更加新颖现代的WPF来进行UI的制作。
WPF的编程语言跟网页的代码很相像,由于之前因为想要美化博客,学了些CSS,所以感觉上手还算比较容易的。磕磕碰碰写了一晚上,终于大概搞明白了UI是一个什么东西,大概怎么写的。
不过WPF自带的控件也不见得要比WinForm要好到哪里去,所以我又找到了一套Metro风格的自定义控件MahApps,这套控件的设计语言简洁清新,实在是很符合我的口味。
当然,最重要的还是关于地图的显示了。
这里我最开始想过要自己画出地图,因为这样的话地图是矢量的可以更方便的支持放大和缩小,不会失真。而且坐标什么的也比较好定义。不过这真的是超出了我的能力范围,最后权衡取舍,我们还是选用了现成的地铁图的方案。
地铁图有了,不过要怎么展示呢?这么大的地铁图,如果铺满窗口的话会显得非常小,而如果放到1:1大小的话,要是用滚动条进行拖动就非常麻烦。这点困扰了我很久,最后经过查询网上的资料,终于找了了用WPF自带的相对坐标变换组对象,在其中添加了对于鼠标滚轮和鼠标拖动对应的事件,实现了鼠标滚轮缩放和鼠标拖动移动位置。简直跟在线地图中的实现一模一样,虽然说起来简单。后台代码也有一百多行呢。恩这个功能可以吹一下!
再就是对于动画的显示,主要思路是根据后台返回的结果在查询对应站的坐标,然后后台新建响应的控件在地铁站的坐标上。这里对于换乘的站点进行了特殊标注,一共有两种不同的图标。
这说起来简单,但是由于是动画,一定有一定的展示时间,所以就免不了在线程中进行休眠,在这个时候,如果用UI线程去进行这个工作的话,整个界面是停止响应的。这时用户如果想要放大缩小移动地图也完全不可能,所以唯一的做法就是新建一个线程专门负责动画绘制。恩看起来也不怎么难。但是。。。
C#很坑爹啊!只有UI线程才能操作UI啊,子线程操作UI直接崩异常啊!但是UI进程又不能去处理这个动画绘制工作啊!怎么办,我找了很多资料,终于找到了一个叫做backgroundworker的东西,他可以每完成一个任务报告一下进度,然后让主进程来进行一部分处理,然后继续完成下一个任务。这样的话,直接把任务设置为休眠一段时间,然后再提示主进程来画一个控件,然后子进程继续休眠,如此循环直到绘制完成。这才把我的问题解决。同时,我还提供了清除路径按钮,如果正在绘制动画的时候按下就会停止当前动画并清除所有之前的路径,当然绘制完成以后清除也是可以的。
再说另一个坑爹之处,之前也讲到我的遍历全图算法要进行一分多钟,而在这个时间总不能让用户傻等着吧。所以我希望能够后台执行查找算法的同时,前台能够继续进行其他的路径计算,等到获得了查找结果再通知用户并显示出来。这里就用到了异步的概念。好在C#很强大的提供了await这个关键字,能够在执行到此时直接返回,等到获得了结果再执行await下面的语句,这简直是神器啊!用上了这个功能,我的程序就能同时执行两个查找了,一个在前台执行需要时间短的长度和换乘优先查找,然后把结果展示给用户,然后后台执行耗时长的遍历全图查找,最大效率的节省用户的时间,在查找算法效率很难提高的前提下尽可能的提升用户体验。恩这可以吹一年!
然后呢在UI中我也尽力添加了一些小功能,比如切换动画速度啊,关闭动画啊,切换起点站终点站啊,尽量做到完美。
原文:http://www.cnblogs.com/shhr/p/5927994.html