设计的载体是什么

本贴最后更新于 2641 天前,其中的信息可能已经斗转星移

也是一篇发在公司内刊上的文章。

从参加工作开始,我就一直困惑于如何写设计文档。公司的各种规定和规范看起来很美,但真的执行起来却总让我头痛不已;同时,我也觉得自己喜欢的方式有些自由散漫,满足不了商业化开发的要求。直到前不久,偶然在网上看到了一篇题为《源代码就是设计》的文章后,我终于感觉自己把这个问题想得清楚些了,所以写了本文,与大家交流。这也许是篇偏激的文章,但我相信它提到的很多问题是我们难以回避的。

公司一直把文档作为设计的主要部分,并且,我认为,公司对这份文档的要求至少包含以下几点:

  1. 能够指导程序员编码。最好是设计师可以完全脱离编码,完成设计后能立即投入下一个项目。
  2. 提供评审和 QA 监督的依据,以便尽早发现不合格项,尽早纠正。
  3. 作为项目维护的依据。

但仔细想想会发现,文档并非做到这几点的必要条件。换句话说,有了文档不见得能做到,没有文档也不见得做不到。

首先,只要设计没有细化到本身就是代码的程度,程序员就不可能仅参照设计编码。这是因为,文档——即使是写作过程中使用了很多软件设计工具的文档——也是一种更接近自然语言的东西,而自然语言有一个天生的弊端——二意性。所以,不论它写的有多好,程序员理解起来也不会与设计师所想的完全一致。而让设计师完全脱离编码也是一个不太现实的想法,一个设计师或许能在一个项目中这样做,但如果每个项目是都是如此的话,他日后肯定会眼高手低,那时他的设计就根本没法用了。我们也许听到过一些新闻,说国外某项目的主设计师根本没写过代码。我不否认这些新闻,但我认为我们绝不能照搬这种模式,因为那些设计师往往是某方面的世界级权威,而人比人是要死的。

其次,评审和 QA 方面的目标也基本没有可能达到。因为大多数情况下这两个过程是“外行监督内行”。我这里并不是说参加评审的同事和项目组的 QA 水平不足,只是说相对于项目组成员,他们对项目的投入要少的多,对项目细节的了解也少的多。这种情况下还想让他们提出评审问题或监督项目执行,要求未免高了一些。以 XYZ 产品为例,评审阶段发现的问题数一般不多,到了 Y 项目更是可怜,只有组织级要求的几分之一了。而如果再仔细观察一下这些问题,还会发现其中绝大部分无关紧要,真正价值较大的往往只有一两个。至于 QA,由于其监督的主要是过程,所以更容易糊弄,并且极有可能是项目组的故意糊弄(评审上的一般是无意的)。我相信这种糊弄在各个公司都不罕见,前一段时间来培训单元测试的讲师就讲到过他们从前(应该是在华为)在项目中糊弄 QA 的事例。作为反例,公司的统计数据已经表明,代码走查这个内行监督外行的过程发现问题的效率是最高的。

第三,已有的经验也说明设计文档在项目维护过程中没多大用。从公司内部看,到目前为止我还不知道哪个项目是主要靠设计文档维护的。项目出问题后,我们首先想到的一般是找参与过项目的人,然后才是文档和其它的东西。我想这个过程应该能说明:人,而不是文档,才是项目维护中最重要的因素。当然,我们也可以说这是因为文档写的不够完美,不过我希望你读完本文后能同意“完美的文档是不可能的”这个观点。再者,从我接触过的开源项目看,项目的维护也不需要文档,开源项目一般没有太好的设计文档,但人们还是能仅仅通过源码和注释等就参与进去。另外微软的文档应该是非常好了,可开发时我还是要经常去看看 MFC、ATL 的源代码,因为那才是最准确的第一手资料,MSDN 上描述的不太清楚的东西,看看源码就一目了然了。

其实,我觉得,把文档看成设计不光满足不了公司的要求,还会带来其他的一些弊端,比如工作效率和一致性等。

作为开发人员,我想没有几个人喜欢写文档,我们常说“兴趣就是动力”,那没了兴趣肯定也就没动力了,所以写文档的效率也就可想而知了。以我自己做例子,我认为自己写文档的效率比写代码低的多。即使抛开主观因素,由于中英文输入效率的差别和画图与打字的效率差别,写文档的也要写代码慢不少。所以,一份“完美的设计文档”可能会大幅增加项目成本,以至于不能公司不会接受。

再进一步,就算能写出来一份“完美的设计文档”,我们也不可能一劳永逸,因为设计会变更。大家往往非常关注需求变更,但设计变更发生的更频繁,道理很简单:需求变更会导致设计变更,需求不变更设计也可能变更。设计发生变更的时候,代码或许已经完成了,加上工期的要求等,开发人员很有可能会先改代码后改文档,这时问题就出现了:开发人员会“忘了”改文档。或者即使他没忘,由于这份文档的初稿很“完美”,他要修改的地方会很多,工作量会很大,甚至要花数倍于他改代码所用的时间才行,再考虑上人的惰性、其他工作的压力等因素,他敷衍了事应该是再正常不过的了。而我们前面已经分析过,这种敷衍是很难被发现的。反正不管是因为什么原因,经过变更后,我们的“完美”文档就和代码对不上了。

XYZ 是一个以数据处理为主要功能的项目,所以我在刚开始时考虑了很长时间“在不能保证数据的正确性时应该怎么办”这个问题,最后的决定是把这些数据扔掉,因为我认为“错误的数据不如没有数据”。把这个原则应用到项目上来应该就是:错误的文档不如没有文档,即使这份文档看起来很“完美”。

回到《源代码就是设计》这篇文章上来。实际上,仅仅是它的标题就让我眼前一亮了。从上学时的《软件工程》教科书,到工作后的实际项目,先入为主的“设计是文档”让我一直没有仔细想过设计应该是什么。这篇文章给了我一个思考的机会,并且经过仔细比较后我发现:代码是比文档更好的设计载体,甚至是天生的设计载体。

我们经常把做软件和盖房子做对比,并把程序员比作建筑工人,这也带来了所谓“软件蓝领”的称呼。但我要说这个对比把程序员和建筑工人放在一块是不恰当的,最简单的理由是建筑工人把砖块磊整齐了非常重要,程序员把代码排整齐了则没那么重要(虽然也很有用)。我认为:程序员应该对应建筑设计师,编码应该对应画图纸。你也许会问:这个对比中,磊砖块对应什么?别着急,软件开发中还有一个大家已经见的太多以至于被忽视了的过程——编译,它对应着磊砖块,也就是把设计变成现实的这个过程,只不过由于软件开发的自动化程度太高,它完全被计算机完成了而已。如果我们把盖房子换成加工机器零件,这个对比的正确性就更明显了,现在自控机床已经能自动加工零件了,而以前,这个工作是要很多人工的。或许未来会有一种机器能根据图纸自动盖房子。我们发现了编译这个之前一直被忽视了的过程之后,应该会注意到,在把文档看成设计的那个对比中,盖房子没有与编译对应的过程,这也说明它是不恰当的。

把代码看成设计的一个显而易见的好处是消除了设计和实现的不一致性,编译代码速度很快,时间成本极低并且基本不需要人工,所以保证这二者的一致是非常容易的。另外它也提高了工作效率,因为主观方面开发人员更喜欢写代码,客观方面英文输入速度比中文快得多。

虽然有不少好处,但从文档到代码的这个转变有点太大,恐怕大家会非常怀疑它是否真的可行。下面我就来分析一下大家可能的疑问。

第一个问题是设计师和程序员如何协作,也就是说设计师怎么让程序员按照自己的想法去做事情。我们先来看设计到底是怎么进行的,乍一看大家可能认为设计是“想”出来的,但实际上它更多的是“试”出来的。任何一个项目都会有一些设计师之前没遇到过的问题,区别只是多少而已。对这些问题,设计师可能会在脑子里想一些方法,然后他就要去试验这些方法是否真的可行,而试验成功之时,也是针对这些问题的设计完成之时,并且,往往也是相关代码写完之时。基于这一点,我认为设计师在一个项目中的工作是完成系统框架和技术难点的开发,程序员则去做具体细节的开发。细节中的一些问题可能是程序员没有遇到过的,解决这些问题的过程,不管是完全靠自己还是求助于设计师,对程序员来说也是一个做“设计”的过程。所以,这种模式下,是不必严格区分设计师和程序员的,大家都是设计师也都是程序员,而这也正好符合“源代码是设计”这一观点。

第二个问题是如何审查监督。我的看法是把这项工作交给项目经理或主设计师,而不做严格的外部审查和监督。自己监督自己也许听起来不那么可行,但我相信大多数人,尤其是项目经理和主设计师,对项目是负责的,所以这样做效果不一定就不好。用人不疑,疑人不用,这句老话用在这里应该很合适。而且,我们前面已经分析过,如果项目组成员成心欺骗,外部监督机制也不见得有多大效果。

第三个问题是项目的风险会不会增大。设计做的不好一直被认为是项目拖期的主要原因之一,现在这个(按原来的观点)看起来根本没有设计的模式会不会加大这个风险呢?我觉得不会,因为它把编码提前了,而编码阶段是最容易发现问题的阶段,所以设计上的问题会被更早的发现,更早的修正,从而降低项目风险。另外,代码重构(在这个模式中相当于设计变更)的代价可能也没有想象的那么大,我自己经常重构大量代码,我觉得花的时间并不多。最后说句实话,XYZ 相关的项目一直都是先编码后文档,但它并没有因此发生过拖期,而且从公司的统计数字看,它的生产率也一直是高于组织级标准。

第四个问题是项目如何维护。前面我已经论证了文档不可能作为项目维护的依据,或者至少是主要依据,所以没有文档也不会让这个情况再变坏多少。另外,不做开发的人可能会觉得文档更易读,但开发人员就不是这样了,他们受过专业训练,对代码的条件反射更强烈,就像音乐家看到 123 会首先想到哆来咪一样。因此,项目维护是可以基于代码进行的,而且,稍后我会说明这种新的开发模式也并非没有任何文档。

前面的文字应该已经基本证明了“源代码就是设计”的可行性,但一个项目如果只有代码肯定也不行,因为代码的组织很松散,开发人员很难通过它们形成对系统的总体认识。所以,一些文档是必须的。下面我就对“应该有哪些文档”和“这些文档应该怎么写”发表一下看法。

第一份应该有的文档是需求,虽然它是需求阶段的输出,但我也要在这里强调它。我认为需求是一个项目中最重要的文档,它规定了系统开发出来后应该是什么样的,必须全面而详细。有经验的开发人员往往看到需求就知道“应该如何实现这个系统”或者“这个系统应该是怎么实现的”了。

第二份文档是系统的总体结构图和一些简要说明,它应该在系统框架开发完成后输出,以便指导项目的后续开发。这份文档可能只要很少的几页就够了。

第三份文档是通讯协议格式和文件格式等,这些内容是结构性的,而不是逻辑性的,所以文档能更清晰的把它们描述出来。数据库格式也可以写,但考虑到可以在大多数 DBMS 中直接看到数据库结构,它的必要性并不大。这些文档可以在相关部分开发完毕后输出,也可以等整个项目开发完了再输出。

第四份文档一些描述系统关键点实现方法的论文。由于针对单个问题,所以它们的主题很明确,不大可能因为太松散而失去价值。这些论文可以等项目结束后再写。

总之,我认为设计相关的文档应该等代码完成之后再写,这样最大程度的避免了文档和代码的不一致。这些文档的篇幅也不宜过大,把一些框架性的东西说清即可,描述具体细节是代码的事。另外我觉得公司也可以考虑为项目组或个人建立内部 BLOG,并鼓励大家发表相关文章,以促进交流。

  • B3log

    B3log 是一个开源组织,名字来源于“Bulletin Board Blog”缩写,目标是将独立博客与论坛结合,形成一种新的网络社区体验,详细请看 B3log 构思。目前 B3log 已经开源了多款产品:SymSoloVditor思源笔记

    1063 引用 • 3455 回帖 • 149 关注
  • 软件工程
    29 引用 • 81 回帖
  • 文档
    56 引用 • 1289 回帖 • 2 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • Sphinx

    Sphinx 是一个基于 SQL 的全文检索引擎,可以结合 MySQL、PostgreSQL 做全文搜索,它可以提供比数据库本身更专业的搜索功能,使得应用程序更容易实现专业化的全文检索。

    1 引用 • 229 关注
  • Git

    Git 是 Linux Torvalds 为了帮助管理 Linux 内核开发而开发的一个开放源码的版本控制软件。

    211 引用 • 358 回帖
  • Solidity

    Solidity 是一种智能合约高级语言,运行在 [以太坊] 虚拟机(EVM)之上。它的语法接近于 JavaScript,是一种面向对象的语言。

    3 引用 • 18 回帖 • 438 关注
  • 黑曜石

    黑曜石是一款强大的知识库工具,支持本地 Markdown 文件编辑,支持双向链接和关系图。

    A second brain, for you, forever.

    24 引用 • 242 回帖
  • 30Seconds

    📙 前端知识精选集,包含 HTML、CSS、JavaScript、React、Node、安全等方面,每天仅需 30 秒。

    • 精选常见面试题,帮助您准备下一次面试
    • 精选常见交互,帮助您拥有简洁酷炫的站点
    • 精选有用的 React 片段,帮助你获取最佳实践
    • 精选常见代码集,帮助您提高打码效率
    • 整理前端界的最新资讯,邀您一同探索新世界
    488 引用 • 384 回帖 • 5 关注
  • 思源笔记

    思源笔记是一款隐私优先的个人知识管理系统,支持完全离线使用,同时也支持端到端加密同步。

    融合块、大纲和双向链接,重构你的思维。

    26030 引用 • 108021 回帖
  • LaTeX

    LaTeX(音译“拉泰赫”)是一种基于 ΤΕΧ 的排版系统,由美国计算机学家莱斯利·兰伯特(Leslie Lamport)在 20 世纪 80 年代初期开发,利用这种格式,即使使用者没有排版和程序设计的知识也可以充分发挥由 TeX 所提供的强大功能,能在几天,甚至几小时内生成很多具有书籍质量的印刷品。对于生成复杂表格和数学公式,这一点表现得尤为突出。因此它非常适用于生成高印刷质量的科技和数学类文档。

    12 引用 • 59 回帖 • 5 关注
  • Excel
    31 引用 • 28 回帖 • 2 关注
  • CSDN

    CSDN (Chinese Software Developer Network) 创立于 1999 年,是中国的 IT 社区和服务平台,为中国的软件开发者和 IT 从业者提供知识传播、职业发展、软件开发等全生命周期服务,满足他们在职业发展中学习及共享知识和信息、建立职业发展社交圈、通过软件开发实现技术商业化等刚性需求。

    14 引用 • 155 回帖
  • 反馈

    Communication channel for makers and users.

    120 引用 • 906 回帖 • 280 关注
  • JRebel

    JRebel 是一款 Java 虚拟机插件,它使得 Java 程序员能在不进行重部署的情况下,即时看到代码的改变对一个应用程序带来的影响。

    26 引用 • 78 回帖 • 675 关注
  • SendCloud

    SendCloud 由搜狐武汉研发中心孵化的项目,是致力于为开发者提供高质量的触发邮件服务的云端邮件发送平台,为开发者提供便利的 API 接口来调用服务,让邮件准确迅速到达用户收件箱并获得强大的追踪数据。

    2 引用 • 8 回帖 • 504 关注
  • LeetCode

    LeetCode(力扣)是一个全球极客挚爱的高质量技术成长平台,想要学习和提升专业能力从这里开始,充足技术干货等你来啃,轻松拿下 Dream Offer!

    209 引用 • 72 回帖 • 1 关注
  • Vim

    Vim 是类 UNIX 系统文本编辑器 Vi 的加强版本,加入了更多特性来帮助编辑源代码。Vim 的部分增强功能包括文件比较(vimdiff)、语法高亮、全面的帮助系统、本地脚本(Vimscript)和便于选择的可视化模式。

    29 引用 • 66 回帖 • 4 关注
  • 音乐

    你听到信仰的声音了么?

    62 引用 • 512 回帖
  • SOHO

    为成为自由职业者在家办公而努力吧!

    7 引用 • 55 回帖
  • sts
    2 引用 • 2 回帖 • 243 关注
  • HTML

    HTML5 是 HTML 下一个的主要修订版本,现在仍处于发展阶段。广义论及 HTML5 时,实际指的是包括 HTML、CSS 和 JavaScript 在内的一套技术组合。

    108 引用 • 295 回帖
  • RESTful

    一种软件架构设计风格而不是标准,提供了一组设计原则和约束条件,主要用于客户端和服务器交互类的软件。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。

    30 引用 • 114 回帖 • 1 关注
  • FlowUs

    FlowUs.息流 个人及团队的新一代生产力工具。

    让复杂的信息管理更轻松、自由、充满创意。

    1 引用 • 6 关注
  • Ubuntu

    Ubuntu(友帮拓、优般图、乌班图)是一个以桌面应用为主的 Linux 操作系统,其名称来自非洲南部祖鲁语或豪萨语的“ubuntu”一词,意思是“人性”、“我的存在是因为大家的存在”,是非洲传统的一种价值观,类似华人社会的“仁爱”思想。Ubuntu 的目标在于为一般用户提供一个最新的、同时又相当稳定的主要由自由软件构建而成的操作系统。

    127 引用 • 169 回帖 • 1 关注
  • Laravel

    Laravel 是一套简洁、优雅的 PHP Web 开发框架。它采用 MVC 设计,是一款崇尚开发效率的全栈框架。

    19 引用 • 23 回帖 • 737 关注
  • Lute

    Lute 是一款结构化的 Markdown 引擎,支持 Go 和 JavaScript。

    29 引用 • 202 回帖 • 28 关注
  • CodeMirror
    2 引用 • 17 回帖 • 166 关注
  • MyBatis

    MyBatis 本是 Apache 软件基金会 的一个开源项目 iBatis,2010 年这个项目由 Apache 软件基金会迁移到了 google code,并且改名为 MyBatis ,2013 年 11 月再次迁移到了 GitHub。

    173 引用 • 414 回帖 • 363 关注
  • etcd

    etcd 是一个分布式、高可用的 key-value 数据存储,专门用于在分布式系统中保存关键数据。

    6 引用 • 26 回帖 • 541 关注
  • SQLite

    SQLite 是一个进程内的库,实现了自给自足的、无服务器的、零配置的、事务性的 SQL 数据库引擎。SQLite 是全世界使用最为广泛的数据库引擎。

    4 引用 • 7 回帖 • 5 关注