我的 10 年自学编程之路
作者 | 码农网–小峰翻译
声明 | 本文是 码农网–小峰翻译 原创,已获授权发布,未经原作者允许请勿转载
为什么每个人都这样匆忙?
走进任何一家书店,你都能看到诸如《24小时自学Java》这样的书,可能书名中的 Java 会变成 C、SQL、Ruby、算法等,然后 24 小时会变成 n 天或 n 小时。在亚马逊高级搜索[title: teach, yourself, hours, since: 2000 ,你会发现有 512 种这样的书籍。排在前 10 位的,9 个是编程类的书(还有一个是关于财务会计的)。还有多种变化,例如“学习”替换“自学”,“天”替换“小时”。
结论:要么大家都特别匆忙地在学习编程,要么编程在某种程度上令人难以置信地比其他任何东西都更容易学习。Felleisen 等人在他们《How to Design Programs》这本书中对这种趋势表示了赞同——“写出不好的程序很容易。傻瓜也可以在 21 天时间内学习编程。即使他们毫无基础。”
让我们来分析一下,类似于《24小时自学C++》这样的书意味着:
自学:在这 24 小时内,你不会有时间写一些有意义的程序,因此不能从中总结成功或失败的经验教训。你不会有时间和有经验的程序员工作,因此不知道生活在一个 C ++ 环境中是什么样的。总之,你不会有时间学到太多东西。这本书只会触及一些表层的东西,不会让你有一个深入的理解。正如 Alexander Pope 说的那样,浅学误人。
C ++:在这24小时内,你也许可以学到一些 C ++ 的语法(如果你学过另一种语言的话),但你无法学到该如何使用这语言。简而言之,假设你是一个 Basic 程序员,那么你可能会用 Basic 语言的风格去编写 C ++ 语法程序,也就是说你不会真正明白 C ++ 真正的优势(和劣势)。那么,这还有什么意义呢?Alan Perlis 曾经说过:“如果编程语言不能影响你的编程思维,那就不值得去学”。一种可能的情况是,你必须学习一点 C++(或者更可能的,是 JavaScript 或进程处理),因为你需要用来与现有的工具配合完成特定任务。但此时你不是在学习如何编程:你学习的是如何完成任务。
24小时:不幸的是,这是不够的,具体原因下面会讲到。
10年自学编程
研究人员(Bloom (1985), Bryan & Harter (1899), Hayes (1989), Simmon & Chase (1973)) 表明,我们大约需要 10 年的时间来发展任何一种广泛领域的专业知识,例如国际象棋,音乐作曲,电报操作,绘画,钢琴,游泳,网球,以及神经心理学和拓扑学等。思考性的实践是关键:不要只是一遍又一遍地做,而是要用超出你现有能力的任务挑战自己,尝试它,在你做的时候和做完之后分析自己的表现,并纠正错误。然后重复,再重复。这里没有真正的捷径:即使是 4 岁时就已经享誉为音乐神童的莫扎特,也依然花了 13 年的时间才开始做出世界一流的音乐。
Malcolm Gladwell 通俗化了这个思想观点,虽然他专注编程 10,000小时,而不是 10 年。Henri Cartier-Bresson (1908-2004)还有另外一个指标:“你的前一万张照片正是你最糟糕的作品。”(他没有考虑到数码相机,拥有数码相机使人在一周内达到这一标准。)真正的专业化可能需要一生的时间:Samuel Johnson (1709-1784)说,“只有通过一生的努力才能在某一学科取得卓越成就:打了折扣的就学不到家。”以及 Chaucer (1340-1400) 曾抱怨说:“生之有限,学也无涯。”Hippocrates (c. 400BC)更是以“生命短暂,艺术长远”的语录而闻名。因此,假设所有技能(如编程,下棋,玩跳棋,玩音乐)都可以在相同量的时间内掌握,是不合理的,而且每个人所需的时间也不会完全相同。正如 K. Anders Ericsson 教授所说的那样,“在大多数领域中,表明即使是最有才华的人也需要多少的时间才能达到最高的性能水平的言论,是值得关注的。10000 个小时这个数值只是给你一个概念而已,因为每个人的学习能力也不尽相同。“
所以,你想成为一个程序员吗?
下面是我编程成功秘诀:
对编程感兴趣,然后因为它有趣而去写程序。请确保保持足够的乐趣,这样你才会心甘情愿地花费 10年/ 10,000 小时在编程上。
编程。最好的学习方式是边做边学。用更技术化的说法就是,“对于个人,在给定领域中最高层次的表现,不会因为经验的增加而自动获得,但是性能水平可以由于经验丰富的个人改善有意识的努力而增加。(p 366)” 还有,“最有效的学习需要明确的任务,针对特定个体的适当难度水平,信息反馈,以及重复和改正错误的机会。” (p 20-21)。对于这个观点,可以参考《Cognition in Practice: Mind, Mathematics, and Culture in Everyday Life》这本书。
与其他程序员交谈;阅读其他程序。这比任何书本和训练课程都更重要。
如果你想的话,可以花四年时间去读大学(或更多时间去读研究生)。上大学使得你有机会接触到一些需要学历证明的工作,而且能够更深入地了解这个领域,但如果你不喜欢学校的话,你也可以(通过一些贡献)获得相似的工作经历。在任何情况下,单靠书本知识是远远不够的。“计算机教育并不比学习更能造就程序专家,就像颜料不能使人成为绘画专家一样。”Eric Raymond,《The New Hacker’s Dictionary》的作者这样说道。我曾经聘用的最好的程序员之一就只有高中学历——他制作了许多优秀的软件,有他自己的新闻组,并且在股票期权上赚到了足够多的钱买下属于他的酒吧。
和其他程序员一起做项目。在一些项目上成为最好的程序员;在一些其他的项目上则是最差的程序员。当你是最好的程序员的时候,你需要测试自己的能力来带领项目,并且用你的观点激励他人。当你是最差的程序员的时候,你需要向大师学习,并且学习去做那些他们不喜欢的事情(因为他们会让你为他们做这些事)。
在其他程序员之后接手项目。理解别人写的程序。知道如何理解和修复程序,当初始程序员不在身边的时候。思考如何设计你的程序使其更容易在你之后被人维护。
学习至少半打编程语言。包括一种注重类抽象的语言(如 Java或 C ++),一种注重函数抽象的语言(如 Lisp 或 ML 或Haskell),一种支持语法抽象的语言(如 Lisp),一种支持声明性规范的语言(如 Prolog 或 C ++ 模板),以及一种注重并行的语言(如 Clojure 或 Go)。
请记住,“计算机科学”中包含“计算机”。知道你的计算机需要多长时间才能执行一条指令、从内存中(有和没有缓存未命中)读取一个单词、从磁盘中读取连续的单词、以及搜寻磁盘上的新位置。(答案在这里。)
涉足一种语言的标准化工作。可以是 ANSI C ++,也可以是决定自己团队的编码风格是否要有2个或4个空格的缩进。无论哪种方式,你会了解到其他人在某种语言中喜欢什么,程度有多深,甚至一些关于为什么他们这么觉得的原因。
有良好的意识能够尽可能快的脱离语言标准化工作。
知道了上面这些,你会质疑书本学习究竟能帮助我们走多远。在我第一个孩子出生前,我阅读了所有《如何……》的书籍,但仍然感觉自己像个手足无措的新手。30 个月后,当我第二个孩子出生的时候,我又再次回到书本中复习了吗?不,相反,我依赖的是我的个人经验,而这其实比那些专家撰写的成千上万页的书本更为有用的多,也更让我放心。
Fred Brooks,在他的随笔《No Silver Bullet》制定了一个三步走的计划,用来寻找优秀的软件设计师:
系统化地尽早识别顶级的设计师。
分配职业导师负责未来的成长,并认真保存其职业文档。
为设计师的互动和相互促进成长提供机会。
这是假定一些人已经具备了成为一个伟大的设计师所需要的品质;工作不过是引导他们前进。Alan Perlis 说得更简洁:“我们可以教每一个人去雕刻:米开朗基罗也曾被教导不要怎么样。伟大的程序员也是如此。”Perlis 的意思是说,伟人所拥有的一些内在品质,超越了训练。但是,这些品质从何而来?它是与生俱来的?亦或者是通过勤奋而开发的?正如 Auguste Gusteau(《Ratatouille》电影中的虚构厨师)所说的那样,“任何人都可以做饭,但只有无畏者才能成就伟大。”我觉得这里的无畏更像是愿意奉献自己生活中很大的一部分时间到思考实践中的意思。可能无畏是对此的总结。或者,正如 Gusteau 的评论员,Anton Ego 说的那样说:“不是每个人都能成为伟大的艺术家,但是伟大的艺术家可以来自任何地方。”
所以尽管去买那些 Java / Ruby/Javascript / PHP 的书;你可能将会从中得到一些用处。但它们改变不了你的生活,你也不可能在 24 小时或 21 天内成长为一个真正全面化的专业程序员。千里之行始于足下,从现在开始努力吧……
经验分享