`
zhmocean
  • 浏览: 202417 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

今天还是明天――一个人的软件工程

阅读更多

到底为什么要用软件工程,这个话题有些大,有时候实在搞不清楚别人是在考自己还是他们真的想了解,经验的微薄,使得每每被人问起的时候,手都不由自主的伸向那本《软件工程》而后又若有所失的放下它――项目中没有遇到困难,那书上所述并无法回答这个看似简单的问题。<o:p></o:p>

而即便是勉强拿出自己的些许心得向大家解释之后,也往往得到一个让人无奈的结论:“我一个人做项目用不到软件工程”――实在不清楚目前这样的作坊式工程还有多少,既然现实无法改变,好,那我们今天就来说说这个问题――一个人的软件工程,当然这里使用了“软件工程”这个沉重的词汇,似乎有点哗众取宠之嫌,不过在没有找到合适的词之前,姑且先容忍我借用一些概念。<o:p></o:p>

前提和事实

按一般习惯,描述一件事情,都应该有前提来约束,本文的前提如下:<o:p></o:p>

1、             既然是一个人的软件工程,那么这个前提很简单:软件的开发完全由你一个人来负责(这种开发方式应该符合国内大多数软件公司的现状)。<o:p></o:p>

2、             用户也并不清楚自己要什么。<o:p></o:p>

3、             你的项目不是外包项目。<o:p></o:p>

4、             你可以正常进行程序编码――起码会写程序吧。<o:p></o:p>

有前提当然得有事实支撑:<o:p></o:p>

1、             3个月后,程序员读自己的代码与读别人的代码并无两样。<o:p></o:p>

2、             用户的需求是一定会变的。<o:p></o:p>

3、             如果项目不考虑成本,就没有风险可言。<o:p></o:p>

4、             代码完整度越高,其变更成本也越高。<o:p></o:p>

5、             风险与需求变化量成正比,需求越不明确,风险越大。<o:p></o:p>

6、             工作量一定的情况下(一般软件项目,工作量只会增加不会减小),降低风险会增加成本,反之亦然。<o:p></o:p>

软件工程的本质

我们可以这样认为,今天的工作叫做成本,而明天的工作叫做风险,软件工程则是为了减少明天的工作,而在今天增加成本的方法学――即为明天而工作。<o:p></o:p>

一般教条式的讲解,都将软件工程分为这几个阶段:分析、设计、编码、测试、维护。姑且不说这种拆分方式太过理想化,单单脱离本质的论述足以让人一头雾水――至少对于很多没有开发经验的人来说是这样――或者说他们只有听的份。所以,我们今天不这样去理解,而是只针对它最实际的目标来讨论,即成本和风险。<o:p></o:p>

就像前面说的一样,成本就是今天的工作,而风险即是明天的工作――未来成本,这听起来似乎是一个简单的加减法:工作量一定,今天干不完可以明天干,保证进度就行了,何必今天去做那么多的事情呢。但我们不要忘记,软件开发是一个特殊的过程,随着项目完成度的增加,其变更成本会越来越大,那么基于这一点,如果今天的成本是1,那么明天做同样工作的成本就是NN>1)。<o:p></o:p>

“明天的事情今天做”、“增加成本”对于我们的老板来说也许都是些心惊肉跳的字眼,这也是很多管理者从简甚至抛弃软件工程的原因,也许这种做法在项目前期根本感觉不到隐患的存在,相反老板(当然也包括程序员)会觉得事情进展的很快很顺利,但很快就会发现,随着需求的变动、人员的调整以及代码的增加,项目所剩的时间会越来越短――而更可笑的是,很多管理者把它归结为资源不足,甚至将其当作加班的理由。<o:p></o:p>

其实如果我们一味的减少今天的工作――即成本,那么明天的工作会很多,这在开发界有个好听的名字叫做“技术债”。过于关注软件局部成本而忽视风险,无疑是在借高利贷――利息可以远远超过本金。笔者曾经使用3天的时间做完一个小型管理系统,期间用户修改了2次,交付后我便再也不愿意去碰那些代码――修改或重构这些代码的时间已经足够我重新编写两个同规模的系统了。<o:p></o:p>

基于这些原理,对于实际开发的项目,尤其是一个人可以负责的项目,那我们的工作无非是两个:今天的设计和明天的变更。<o:p></o:p>

好,说了这么多似乎不着边际的理论,各位也许已经头昏脑胀要扔西红柿了,确实,干巴巴的理论描述没有任何实际体验,着实让人难以下咽,那好,下面我就来重现一个大家都似曾相识的情形。<o:p></o:p>

一个人的项目

一般来说,我们接到一个项目之后会忠诚的按照以下步骤来进行:先做需求分析,清楚用户要什么;然后根据用户的需求,做项目的整体设计;设计完毕,进行编码实现;等软件实现后提交给用户测试确认;用户测试完毕,认可之后,交付使用。<o:p></o:p>

好,如果按照这个过程,说明你已经是一个合格的学生――对,是学生,刚刚毕业的学生水平不过尔尔,因为学校仅仅会教给你事情在一切顺利的情况下怎样去做――我想一个智商没有问题的人都能够做到,这是软件工程最基本的要求。<o:p></o:p>

但事实并不是我们想象的那样,很多时候,一个人的项目很少按照这个过程走,理由很简单:不需要。即便是很多公司在强制进行规范化管理,但那份看似完整的配套文档早就在交付产品之前就失去意义了――它占用了我们太多的时间。<o:p></o:p>

开始前的准备

无论多么多么急,对于一个项目来说,最先做的工作应该是要了解用户需要什么――这个时候,怎么去做似乎成了次要的。但是,基本上每次与用户的接触,我们都会或多或少的失望而归,“这个也许就是XXX那样吧”、“咳~你就看着做吧,跟XX差不多就行”、“这块应该是这样的吧”,这些让人头昏眼花的所谓需求,我想各位听得不会比我少。<o:p></o:p>

的确,对于终端用户的项目,我们很难从他们口中得到非常准确的描述――摸石头过河应该是“客户就是上帝”这句话最好的借口了。没有办法,我们只能将希望寄托在马上就要进行的、激动人心的设计工作。<o:p></o:p>

等等!没有计划么?传统的软件工程可是有分析和计划的啊。不过,那个声音似乎又响起来了“太耗费时间了”、“系统很简单,我们不需要”、“不清楚的时候我们可以问上帝”。<o:p></o:p>

今天的工作

抛开一切繁文缛节,我们终于可以进行设计了!但现在,我们看得见摸得着的却只有今天了,那么今天的工作就是我们最最关心的――成本。既然工作总量不会减少,那我们只需按照理想的计划,做自己份内的工作就行了――我想,这一天的工作我们都能够保证,因为计划对于今天来说是合理的,甚至如果你的速度足够快,大可结余出时间来做别的事情。<o:p></o:p>

但事情似乎并不会按照我们想象的那样发展,因为每天除了这些额定工作,我们还有更多的事情要做。<o:p></o:p>

莫名的错误

或许在你拖着还没睡熟的身躯,刚刚要开始今天的任务的时候,却突然在代码中发现了一个错误,呜呼!更让人恼火的是,这个错误并不是昨天带进来的。<o:p></o:p>

“哈!幸亏我每天都有备份”,我想大多数人都已经给自己留下了后路――现在的压缩软件都很方便,打个包完全是举手之劳。凭借直觉――对,现在能够依靠的只能是直觉――辗转于一个个拥有“规范”命名的压缩包中(或许其中还记录了这个备份都做过什么工作),去寻找这段代码是在什么时候诞生的。<o:p></o:p>

值得庆幸的是,有很多强大的代码比对工具供我们使用,它们为我们节省了大量的时间――有时候,这些工作并不是有时间就能完成的。没用多长时间,我们就找到了错误的诞生地――或许我们还因为使用了折半查找的方法来提高速度,嗯,这是个好的开始。<o:p></o:p>

陌生的代码

在我们因为找到错误代码的诞生地而欢呼之余,一阵冷汗随之而来,我们发现那段代码根本就没有注释,或者使用了大量的所谓的技巧来实现的,或许是一段冗长的意大利面条式代码,也或许是更让人恼火的事情:错误的注释!。哦对,之所以这样,是因为我们的时间有些紧,为了赶工才会这样做的;也可能是因为工作到太晚写出了一些混乱的代码。<o:p></o:p>

不过这个时候,我们没有必要责怪自己的记忆力,也并不是老年痴呆的前兆,转战于如此庞大的代码结构中,或者“我的代码别人没法修改”是当前作坊公司一贯的作风。不过但我们却忽略了这样一个道理:3个月后,你也是“别人”。那么如果真的这样去想,3个月后我们仅凭代码就能发现它的错误已经非常不容易了――当然,发现和修改是两码事。<o:p></o:p>

虽然修改可能如此之难,以至于早餐也即将变成午餐,但问题终归在我们的艰苦努力下解决了,值得庆祝。编译运行……我们可能发现了让人呛水的事情――同样的错误不只一处。<o:p></o:p>

双胞胎?多胞胎?

按照业界公认的理论,吭哧C和吭哧V(复制和粘贴)方法是速度最快的,发明这个功能的人或许应该获得诺贝尔工业奖,因此这一方法被众多程序员当作提高效率的法宝。我们应该也不例外,点两下鼠标就会出现一个复制品的功能简直太具有诱惑力了,不过,我们似乎还发现了它的一个致命缺陷――即便是错误,它也会原封不动的克隆一个。呜~那我们刚才发现的问题可能就是来自于此。<o:p></o:p>

或许这对于我们来说也并不可怕,既然第一段代码已经修正,那么将它再克隆几次不就可以了吗?为了午饭,说干就干!<o:p></o:p>

彩票式设计

匆匆吃完午饭,忐忑不安的跑回办公室――刚刚修改的代码还没有测试呢。编译、运行,出错。应该是(嗯,很流行的口头禅)刚才一时疏忽写错了代码。再经过一番调试,代码终于通过了――注意,是我们自己认为通过了。<o:p></o:p>

之后又是一阵复制、粘贴,问题终于算是解决了,当然我们自己是这样认为的,别人问起,也只能回答:应该吧。这样的心安理得对于我们这个小项目来说已经足够了。或许到这里我们还不能彻底松口气,因为错误的来源是一个礼拜之前,那么这就意味着,自那天来的所有备份文件都需要修正这个错误――同样是传统的方法。而至于这些修改是不是都起作用了,呼!上帝才知道,我的时间不够去一一验证了,就当买了张彩票。<o:p></o:p>

花了半天加上一中午的时间,嗯,还好,只用半天的时间就解决了发现的错误。我们终于可以在后面半天的时间里进行下一步的设计工作了――对于除错来说,这一天的设计工作简直太简单了,几乎用不了半天。<o:p></o:p>

用户的实验田

午休时间应该也过了,我们应该打开设计要求,准备下一步的工作了。但就在此时,项目经理(这个名词似乎很流行,也很光亮,但我实在无法将其跟一个人的项目联系起来,或者它仅仅是某个词的文明用语:包工头)突然跑过来(对!他只带来了一张嘴),说用户传话过来(怎么感觉像那个?难道是我古装戏看多了?),他想在现有的系统上增加一些功能,但这些功能他只是想看看效果,如果效果好,就可以真正增加进系统――这种“无礼”要求也许已经让你暴跳如雷了,我们成了小白鼠!<o:p></o:p>

不过仔细想想,我们不是一直在这样安慰自己吗?“用户就是上帝”,嗯,小白鼠就小白鼠了,而且实际上这个问题也不难解决,复制出一份工程文件来不就OK了?<o:p></o:p>

项目经理继续说:用户要的功能并不仅仅这一个,他想试验的还有这个、这个、这个……。好吧,尽管来吧,反正几份都是同样的复制操作。<o:p></o:p>

想不到,项目经理的话还没有完:“上周让你试验的那几个功能,用户比较满意,他要求把它们都加进来……”,如果不出意外,我们的头部会明显感觉到异样,甚至越来越怕看到那个采蘑菇的小姑娘。<o:p></o:p>

“这根本没有在计划之中!”我们终于忍不住叫出来。但是事实就是这样,我们根本没有办法预知这些事情――确切的说是准确的预知。现在需要做的就是硬着头皮往上冲。<o:p></o:p>

当晚餐的香味飘过来的时候,我们可能很幸运的将这些功能都集成在了一起――但仅仅是能用,能用而已。到底有多能用?没有人告诉我们。还好项目经理下班了,用户睡了……好吧!没人来打扰了,那就提提精神来做今天应该做的工作。<o:p></o:p>

我想这种滋味,各位都或多或少或者一直在品尝着,办公室通明的灯光似乎只带来一个奢望:周末我一定要好好的睡个懒觉,我要报复加班的生活……<o:p></o:p>

明天的工作

拖着疲惫的身躯回到床上,已经没有任何力气再去回忆一天的工作,只是期望明天的工作会有些变化,但事实却总是跟期望相反――外甥打灯笼,照旧。希望似乎是在更远的后天。

某天的工作

突然你在某天发现了跟往常不一样的事情――或许这会给你带来一丝新鲜感。<o:p></o:p>

Ÿ           设计的功能根本不是用户想要的;<o:p></o:p>

Ÿ           压缩包的备份数量已经到达了三位数;<o:p></o:p>

Ÿ           数据结构需要变动,但逻辑已经写死在了SQL语句里;<o:p></o:p>

Ÿ           一个月之前解决的问题又冒了出来;<o:p></o:p>

Ÿ           新增加的功能只有部分能够使用;<o:p></o:p>

Ÿ           想改动一处代码,却除了靠眼睛,没有任何理由告诉自己它带来的影响。<o:p></o:p>

Ÿ           修改了1个错误,却带来了10个错误;<o:p></o:p>

Ÿ           用户需要知道这个答案:你凭什么说自己的系统已经满足我的要求,而且没有任何错误?<o:p></o:p>

Ÿ           ……<o:p></o:p>

交付期限越来越近,棘手的问题却越来越多,这些所谓的新鲜感告诉我们,事情似乎已经糟糕到无可救药的地步了,我们开始抱怨用户的要求有多不合理、老板的压榨有多么的狠、程序员是多么的难当、技术是多么的难搞,甚至我们开始憧憬做管理和做业务远比做技术要好。我们错在哪里呢?我们一直辛勤的努力,一刻也没有停歇为什么还是这样的结果呢?<o:p></o:p>

一些建议

由于时间和本人水平的限制,我只是简单的描述了几个具有代表性的细节问题――不要小看这些细节,它们极有可能是堤之蚁穴。<o:p></o:p>

能够坚持读到这里的各位,也许已经隐隐约约在其中找到了自己的影子,甚至有的人会说这是自己的翻版;当然,也会有一些朋友已经解决了这些问题――这是我们最希望看到的结果。问题暴露了,我们并不是搬只马扎来隔岸观火,一起来总结一下其中的经验,或者借来别人的经历对下一个项目给出一些建议,也算是我们的项目没有白白的受尽创伤。<o:p></o:p>

那么,我想这个时候最应该先做的事情就是抛弃“一个人的项目不需要软件工程”的想法,因为它自始至终都在阻碍我们的进程――这个想法让你只关注今天的成本而忽略了明天的风险。通过这个相对典型的项目,我们至少可以认识到以下几点:<o:p></o:p>

Ÿ           抛弃“提高效率就是加紧干活”的观念;<o:p></o:p>

Ÿ           无法预知不等于无法准备,也不等于无法减轻由此带来的影响;<o:p></o:p>

Ÿ           今天的额外成本来自于昨天的风险,而且这个转换超过了11<o:p></o:p>

Ÿ           用户的需求变化应该可控,最起码也是在一定的阶段相对稳定;<o:p></o:p>

Ÿ           注重代码注释的规范性,为方便别人阅读而写代码;<o:p></o:p>

Ÿ           对于代码的变更应加入版本管理机制,记录每次变更的日志;<o:p></o:p>

Ÿ           为变化而设计,明天的事情今天做;<o:p></o:p>

Ÿ           建立完善的单元测试环境,使代码的更动有硬性保障;<o:p></o:p>

Ÿ           避免使用复制粘贴式代码重用,应使用模块化设计;<o:p></o:p>

Ÿ           需求、设计、编码可以以任何速度进行,唯独除错才会耗费时间;<o:p></o:p>

结语

业界有大师语:任何一个傻瓜都能写出计算机可以理解的代码。唯有写出人类容易理解的代码,才是优秀的程序员。那么对于一个软件系统来说,我们的工作就不仅仅是让机器正确执行这么轻松了。优秀的设计与蹩脚的设计唯一区别,就在于前者是向人展示它要干什么。而软件工程之于设计(对于一个人单打独斗的项目,设计应该是我们的绝大部分工作),正是保证了这一点。<o:p></o:p>

由于经验微薄、水平有限,本文所述也许仅仅是井中窥天,无缘于大雅之台,但实望以粗糙的个人感受让各位少走些弯路。<o:p></o:p>

分享到:
评论

相关推荐

    中班数学活动认识昨天今天明天PPT课件.pptx

    这篇PPT课件是针对中班幼儿设计的一次数学活动,旨在帮助孩子们理解和区分“昨天、今天、明天”这三个时间概念。活动通过一系列互动和操作,让孩子们在实践中学习和运用时间词汇,培养他们的基本时间管理能力。 1. ...

    自定义时间选择器,今天,明天,后天

    标题“自定义时间选择器,今天,明天,后天”指的是一个特定的实现,允许用户快速选择今天、明天或后天的日期,这种设计可以提升用户体验,因为用户不需要滚动整个日历来设定日期。 自定义时间选择器通常涉及到UI...

    互联网昨天今天和明天

    互联网昨天今天和明天,本文档为PPT课件生成。

    备忘录,可以提示今天和明天的备忘

    标题中的“备忘录,可以提示今天和明天的备忘”指的是一个软件应用程序,它具有为用户提供日常事务提醒的功能,特别是针对今天和明天的待办事项。这种类型的软件设计目标是帮助用户有效管理时间,避免忘记重要的日程...

    新初一开学第一次主题班会课件:努力今天收获明天(共张PPT).ppt

    1. **中考的重要性**:中考作为人生的一个转折点,是进入优质高中的途径,进而为考取好大学铺平道路。初中的学习成绩直接影响到学生的升学机会,特别是初一和初二的成绩对整个初中阶段,甚至一生的成功起到决定性...

    IOSS判断日期是今天,昨天还是明天,时间戳转时间,时间转时间戳

    1,时间戳转换时间;2,时间转换时间戳;3,时间显示(例如:1分钟以内 显示 : 刚刚 ... // 4)当前时间之前或者明天 显示 : 今天 09:30 明天 09:30 // 5) 今年显示 : 09月12日 // 6) 大于本年 显示 : 2013/09/09)

    通达信公式指标源码软件一天稳定2% 今天买 明天卖.doc

    为了更准确地捕捉到市场动向,主图指标中的VAR1变量是基于最近两日收盘价的平均计算而得,而VAR2则是一个更为复杂的复合指数,它结合了21日斜率变化和20日乘以42日的收盘价平均。这两个变量的交叉点,即VAR1和VAR2的...

    中班数学活动认识昨天今天明天.pptx

    在分组操作的环节中,通过设计与日常生活紧密相关的四个任务,孩子们需要将具体事件与“昨天、今天、明天”相对应的日期进行匹配。例如,在气象任务中,孩子们需要根据昨天的天气情况作出判断;在值日生任务中,安排...

    用户界面的今天和明天

    GUI的一个重要特点是采用了**WIMP(Windows, Icons, Menus, Pointing device)**的交互模式: - **Windows(窗口)**:多个应用程序可以在不同的窗口中同时运行,用户可以轻松地在它们之间切换。 - **Icons(图标)...

    昨天今天明天名言.doc

    昨天今天明天名言.doc

    Python获取昨天、今天、明天开始、结束时间戳的方法

    我们通过调用today()函数来得到一个表示今天的datetime.date对象。 接着,为了获取昨天和明天的日期,我们使用了datetime模块中的timedelta类,它代表了两个日期或时间之间的差异。通过传递days参数,我们可以得到...

    深度学习的昨天今天和明天

    ### 深度学习的昨天、今天与明天 #### 标题解读 标题“深度学习的昨天今天和明天”简明扼要地概括了文章的主要内容:深度学习技术的发展历程、当前的应用状况以及未来的发展趋势。 #### 描述解读 描述中的“蛮...

    小品《昨天今天明天》台词剧本.doc

    小品《昨天今天明天》台词剧本.doc

    昨天今天明天(法院入职感悟).pdf

    昨天今天明天(法院入职感悟).pdf

    深入研究Windows内部原理系列之一:Windows的昨天、今天和明天

    【深入研究Windows内部原理系列之一:Windows的昨天、今天和明天】 Windows操作系统是全球最广泛使用的个人计算机操作系统,它的历史可以追溯到1980年代。这个系列将带你深入了解Windows的发展历程,以及其核心原理...

    web的今天和明天

    根据《WEB的今天和明天》一文的分析,当前WEB应用安全面临的挑战主要包括以下几个方面: 1. **系统层面风险**:老旧的服务器软件如IIS、Apache版本过低,缺乏必要的安全补丁,成为黑客攻击的目标。 2. **应用层面...

    DirectX昨天今天明天.pdf

    ### DirectX昨天今天明天 #### 一、DirectX概述 DirectX是一种由微软开发的多媒体编程接口,被广泛应用于游戏开发和个人计算机图形处理中。它包括一系列API(应用程序接口),用于处理与多媒体相关的任务,特别是...

    小河的昨天、今天和明天作文.doc

    在这片生机勃勃的景象中,我们仿佛能看到小河的昨天,那是一个人与自然和谐共存的黄金时代。 然而,随着社会的发展和经济的繁荣,人们的生活方式发生了翻天覆地的变化。科技的进步和工业的兴起带来了对自然资源的...

Global site tag (gtag.js) - Google Analytics